Account created on 8 April 2010, almost 15 years ago
#

Merge Requests

Recent comments

๐Ÿ‡ณ๐Ÿ‡ฟNew Zealand garethhallnz

Thank you for your thoughtful feedback and for testing. I'm really happy to hear the performance improvements were noticeable!
My thought are:

Hashing Algorithm Selection
I completely agree that hash collisions are not a real concern here. The reason I included multiple options was for compatibility. While xxh3 is significantly faster (around 30x faster than md5), it was only introduced in PHP 8.1. Since the module still supports Drupal 9 via 1.0.x, there are likely users running PHP 7.4 or 8.0 who wouldnโ€™t have access to xxh3.

Options:

  1. Automatically use xxh3 if it's available and fall back to md5 when it's not, without exposing this as a user-configurable option. This simplifies the UI while ensuring backward compatibility.
  2. Drop support for 1.0.x so we only need to support PHP 8.1+ and thus know that xxh3 is available.

Cache Lifetime Setting
The primary reason I added this was to allow flexibility for different caching strategies, particularly in setups using reverse proxy caches like Varnish. For example Lagoon documentation https://docs.lagoon.sh/applications/drupal/services/varnish/ explicitly recommends disabling Drupalโ€™s Internal Page Cache when using a reverse proxy, which means relying on alternative caching mechanisms.

In my own use case, Iโ€™ve found that authenticated users often benefit from different cache policies than anonymous users. That said, I see your point about Tailwindโ€™s atomic nature making stale cache entries unlikely.

I think this setting should remain different because if this module depends on the lifetime setting from the Internal Page Cache module then we really so set the dependency array in the info file.

Options:

  1. Default to permanent caching (CACHE_PERMANENT) but allow developers to override it via hook_alter().
  2. Default to permanent caching (CACHE_PERMANENT), remove the cache lifetime setting from the UI but allow an override in settings.php

I personally don't like the above and far prefer the current setup.

Final Thoughts
I completely understand wanting to keep the UI simple, and I agree that too many options can be overwhelming for site admins. However, Iโ€™d argue that most developers using Tailwind in Drupal are already fairly technical, given how much custom Twig and hook-based class injection is often required to make it work well.

Ultimately, I was scratching an itch and didnโ€™t test everything to death, which is why I marked it as experimental. My personal experience has taught me that caching can be a pain in the butt, so we should tread lightly until we are confident that our assumptions are true (Mostly mine :))

That said, if you feel strongly about reducing the UI complexity, I think the best path forward would be:

I would personally suggest creating a version 2 that drops support for 1.0.x. This way, we can ensure PHP 8.1 is used, allowing us to remove the hashing configuration. Since this module doesnโ€™t depend on the Internal Page Cache (nor do I think it should), I believe having the cache lifetime setting in the UI is reasonable.

๐Ÿ‡ณ๐Ÿ‡ฟNew Zealand garethhallnz

I too have run into a similar issue; for me most of out users are authenticated and Drupal's internal page cache module and even the dynamic page cache module does very little to help. Reverse proxy cache like varnish will obviously help but not all sites have access to such systems.

I have built on top of @stjuan627 patch and expanded on his work.

Why This Feature is Needed

During my testing, I observed that while Drupalโ€™s internal page cache works well for anonymous users, performance for authenticated users suffered significantly. On average, page load times for authenticated users were 2-3 seconds slower due to the lack of caching for compiled Tailwind CSS. This is because authenticated users often bypass Drupalโ€™s internal page cache, leading to repeated CSS compilation for every request.

The new caching feature addresses this issue by:

Improving Performance for Authenticated Users:
- By caching compiled CSS, authenticated users no longer experience the overhead of repeated CSS compilation, resulting in faster page load times.
Enhancing Performance for Anonymous Users:
- While anonymous users already benefit from Drupalโ€™s internal page cache, this feature adds an additional layer of optimization by caching compiled CSS separately. This reduces the time required to generate the final HTML response.
Scalability:
- High-traffic sites, especially those with a mix of anonymous and authenticated users, will see significant performance improvements, as the caching mechanism reduces server load and improves response times.

New Features

1. CSS Caching for Improved Performance

Iโ€™ve added a caching layer to store compiled Tailwind CSS, significantly reducing the need for repeated compilation and improving response times for subsequent requests. Key highlights include:

Configurable Cache Settings:
- Enable or disable CSS caching via the theme settings form.
- Set a cache lifetime, with options ranging from 1 hour to 90 days or permanent storage.
- Choose a hashing algorithm for cache key generation (default: `xxh3` for optimal performance).

Automatic Cache Invalidation:
- The cache is automatically cleared when theme settings are saved or when the cache is manually cleared via Drupalโ€™s cache administration page.

Efficient Cache Key Generation:
- Cache keys are generated based on the unique CSS classes found in your content and the active theme, ensuring accurate and efficient caching.

2. Extraction of Unique CSS Classes

To optimize CSS compilation, Iโ€™ve introduced a new method to extract and deduplicate CSS classes from your HTML content. This ensures that only the necessary CSS is compiled and cached, reducing overhead and improving performance.

3. Enhanced Theme Settings Form

The theme settings form has been updated to include new options for managing the caching feature:
- A new section allows administrators to configure caching settings, including enabling/disabling caching, selecting a hashing algorithm, and setting the cache lifetime.

Improvements

Dependency Injection:
- The `HttpMiddleware` class now injects the `CacheBackendInterface` service, enabling seamless integration with Drupalโ€™s caching system.

Code Refactoring:
- Added helper methods (`injectStyleTag()` and `generateCacheKey()`) to streamline the caching and CSS injection process.

Performance Optimization:
- Reduced redundant CSS compilation by leveraging cached results, improving response times for high-traffic sites.

How to Use the New Features

  1. Enable Caching:
    - Navigate to the theme settings form (`/admin/appearance/settings/[your_theme]`).
    - Under the "Experimental features" section, enable CSS caching and configure the cache lifetime and hashing algorithm.
  2. Monitor Cache Performance:
    - Use Drupalโ€™s cache administration page (`/admin/config/development/performance`) to monitor and clear the cache as needed.
  3. Test and Optimize:
    - Test the caching feature in a staging environment to ensure compatibility with your siteโ€™s configuration.
    - Adjust the cache lifetime and hashing algorithm based on your performance and security requirements.

Backward Compatibility

This release is fully backward-compatible with existing configurations. However, the caching feature is marked as experimental, so I recommend thorough testing before enabling it in production environments.

Bug Fixes

- Fixed an issue where redundant CSS compilation could occur for repeated requests.
- Improved error handling during CSS compilation to ensure graceful fallbacks.

๐Ÿ‡ณ๐Ÿ‡ฟNew Zealand garethhallnz

Hello,
Although I still do a lot of Drupal work, I haven't worked with Drupal Commerce in many years, so I simply haven't had a need for it. I guess what I'm saying is that unless the need arises for me personally, I doubt I'll be porting it. Perhaps the Drupal Foundation might consider funding the portโ€”that way, I could fit it into my regular development responsibilities.

๐Ÿ‡ณ๐Ÿ‡ฟNew Zealand garethhallnz

I just encountered this problem too. After some debugging, I discovered that the issue was due to having the "bypass ECK entity access" permission. When you have this permission, the hook_entity_access is never called.

As per /src/EckEntityAccessControlHandler.php#L52

  public function access(EntityInterface $entity, $operation, AccountInterface $account = NULL, $return_as_object = FALSE) {
    if ($this->canBypassAccessCheck($account)) {
      return $this->getBypassAccessResult($return_as_object);
    }
    else {
      return parent::access($entity, $operation, $account, $return_as_object);
    }
  }
๐Ÿ‡ณ๐Ÿ‡ฟNew Zealand garethhallnz

Resolved in 3.0.1 release.
Thank you for your contribution.

๐Ÿ‡ณ๐Ÿ‡ฟNew Zealand garethhallnz

Resolved in 3.0.1 release.
Thank you for your contribution.

๐Ÿ‡ณ๐Ÿ‡ฟNew Zealand garethhallnz

Resolved in 3.0.1 release.
Thank you for your contribution.

๐Ÿ‡ณ๐Ÿ‡ฟNew Zealand garethhallnz

Considering the age of this issue and the fact that Drupal 7 is on it's leg I think it's fair to say its outdated.

๐Ÿ‡ณ๐Ÿ‡ฟNew Zealand garethhallnz

I have tested the patch in #6, and while it does resolve the render array key errors, it also causes the ArrayWidget to be empty, i.e., no items show in the facet block. The issue seems specifically related to the array keys 'values' and 'value' in methods prepare() and generateValues(), respectively.

Here is an updated patch.

๐Ÿ‡ณ๐Ÿ‡ฟNew Zealand garethhallnz

I initially reviewed this as functional and accurate, but upon further consideration, I believe it requires additional refinement.

Imagine you're conducting an auction with a substantial $500 increment and a reserve price of $10,000. While the reserve hasn't been met, the bid increment works well. However, once the bidding reaches the auction reserve, it can become somewhat unfair on bidders .. forcing them to pay $500 everytime

Production build 0.71.5 2024