Below are a list of some of the technical key points of this plugin.
- PHP in the context of a WordPress plugin
- Use of the WP Admin Ajax as well as the
/wp-json
REST API - Nonces for request verification
- Validation and sanitization of input
- Resource formatting and error handling
- WP custom admin menu for storing API tokens
- Using the WPDB API to create, read from, and update a custom DB table specifically for Plagiarism Checker
- Validation, sanitization, and token management using WP standard security functions
- Use of the WP Admin Ajax as well as the
- Object-Oriented Programming Paradigm
- Complimented by design patterns
- Autoloading with Composer and PHP-DI
- PHP 8.3 features: type hinting, constructor promotion
- Testing
- Testing with Pest (and I set up a variation of the WP scaffold plugin-tests)
- Logging
- Logging with Monolog configured to integrate into the
WP_DEBUG
constants
- Logging with Monolog configured to integrate into the
- JavaScript in the context of WordPress
- Using the WP Admin Ajax API to fetch data from the backend
- Error handling for server failures, non-200 responses, empty data, missing API token, and successful requests
- User Interface
- Sticky, toggleable web form available only to logged-in users
- Cross-theme compatibility styling:
- Inherit from WP-supported styles where possible and set neutral fallbacks where themes do not support them
- Responsive design principles and techniques without the use of media queries
- TypeScript used exclusively
- Both functional and class-based approaches (where they made sense)
- Vite as a build tool
- Modern JS/TS including ES6+/ES2023
- Modular imports and exports
- Assets compiled down to single, optimized files
- Tests written with Vite
- Scss
- BEM methodology
- Object-Oriented CSS (lightly used to separate form structure from style)
- Low specificity styles
Below is a brief, dev focused, outline of some of the app design and architecture choices made for this plugin.
This project takes a hybrid approach to CSS. The default is BEM convention via Scss with a light implementation of SMACSS only where absolutely necessary.
However, where it makes sense, Object Oriented CSS is also employed in order to separate some of the more intricate structural components from their styles.
Theme compatibility is a priority as opposed to one particular styling goal.
In short, the CSS for this project should be composable. Avoid the use of global styles where possible. Inherit from WP for the best theme compatibility where possible. Use intelligent fallbacks to styles that will coordinate across the spectrum of designs.
- Output a widget on the sidebar with an input form and submit button
- On submit, send a REST request to genius.com to search for matching song lyrics
- Return that data and output the results below in the form
Things I would like to do next
- Support cross referencing an entire song
- Compare the lyrics from Genius with the song lyrics and develop a similarity score
- Implement better algorithms for "percentage based" plagiarism
- Have different "plagiarism" scores
- More controls for the user to minimize the toggle
- A user can enter "request" mode, highlight text, and on mouse release a rest request is sent
High level
- On form submission an admin ajax request is passed to the BE
- The BE validates/sanitizes and then makes a request to genius.com
- Request is returned, handled, and then sent back to FE
- FE displays output to user
The goal here is to handle our own user errors (no API key, invalid search text, etc) while also handling error responses such as invalid/expired auth token, empty response, etc from the API we're requesting data from.
The BE has a series of checks to ensure that all possible scenarios are handled and returned in a consistent format to the FE. The Resource class is enforcing consistent response formatting in all responses.
On the FE, TypeScript is checking to ensure that all responses from the BE are routed to the correct places and that data inconsistency will cause problems.
- Create a response class for ensuring consistency in delivering responses to the FE
- Create a Resource_Success class and only return Resource_Success|WP_Error from the Song_Controller
- Extend WP_Error and require the error data as an argument (as opposed to just the required message)
High level
- Handles API token input from users and stores them
- Data storage is on a per user basis
- Custom plugin DB tables for easier separation between site and plugin
TLDR: Don't write end to end feature tests unless you really need to. Let's keep an eye on ways to both run tests based on what kind of mocking they need as well as what group they're part of.
- Pest (version 1 to use Yoast WP Testing Utils)
- Yoast WP Testing Utils (provides access to WP_Unit functions)
This project has three main test categories
- Unit
- Integration (Testing 2 or more components together)
- Feature (End to end test)
If possible, try to keep the majority of your tests to Unit and Simulated Integration tests. Full integration tests take more overhead, are harder to run, and harder to troubleshoot. In addition, they are very easy to misuse and can result in tests that have a vague scope and make it difficult to localize errors. These tests should be saved for end to end functionality, for difficult bugs, or areas where failure would be critical.
Folders vs Groups for organizing tests
You can add text within a collapsed section. Use the folders to organize tests by Unit, Integration, and Feature. Use the groups to organize tests by how they need to load.
Run your tests by group, not by folder. For example, every test in the "wp_brain_monkey" group needs Brain Monkey to run, but not every test with Brain Monkey is an Integration test. Maybe a class can only really be tested if we sniff out something it's doing inside a WP core function that we need to mock with Brain Monkey. This is WordPress, we need to be flexible with our categorization here.
TODO: When a user runs tests by Unit, Integration, or Feature folders let's organize the tests so that they rerun the bootstrap and load the required setup automatically.
WordPress Stubs, Brain Monkey, and loading WP Core
In addition, we have three ways to handle WP dependencies. Testing code that doesn't include them, mocking them with Brain Monkey, or loading them as normal and testing with a test DB.
- Unit (WP stubs are loaded)
- Simulated Integration (WP Core can be mocked with Brain Monkey)
- Full Integration (WP Core is NOT mocked and a testing DB is used)
It's important that we run each test with the kind of mocking they need. Ideally, we need a system where we can group the tests by their type (Unit, Integration, or Feature) as well as their mocking needs.
At the moment, I'm organizing the test folders by Unit, Integration, and Feature and adding these groups based on the mocking needs.
(Underscores "_" to not conflict with Pest command line)
- "wp_full" - Load full WP Core
- "wp_brain_monkey" - Mock with Brain Monkey to make assertions
- (default) "wp_stubs" - Load no WP Core and load WP stubs
There's no way to filter these tests at the moment, but we can take care of that once there are enough tests for that kind of dev work to make sense.
Load both WP Core/Stubs and Brain Monkey together
In a perfect world, we would be able to load the WP Core or the WP stubs and then overwrite them with Brain Monkey. Not impossible! However, we would need pluggable functions (the hard part) and then need to load them after Brain Monkey (easy part).
I did some experiments with lucatume/function-mocker. Using this package, we can load all of WP Core or all of the WP stubs and then patch out functions that we want to write assertions against.
I really like this library and use it for writing tests on legacy code to pin functionality in place. However, the current stable version on composer hasn't been updated in 6 years and I decided to not tie this project down with code that was that old.
I sent a message to the dev and would like to pick this idea up again in the future if the library maintenance changes.
For the moment, this is on hold, but getting WP set up with Pest and having a selectable way of choosing between WP stubs, Brain Monkey, and WP Core would be a great project in and of itself.