Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[WIP] Next basset version #143

Open
wants to merge 38 commits into
base: main
Choose a base branch
from
Open

[WIP] Next basset version #143

wants to merge 38 commits into from

Conversation

pxpm
Copy link
Contributor

@pxpm pxpm commented Dec 4, 2024

What's new? (Summary)

  • Changed the default disk to be the basset disk, it points to the /public folder instead of storage, removing the need for symlink.
  • removed the composer script to link the storage
  • added an option to cache urls when dev mode is on.
  • install command now adds public/{basset_path} to .gitignore
  • local files are only "re-cached" in dev mode when their content hash changes
  • introduced "named assets" through a map function
  • added ability to change the package named assets

Removing the symlink and adding .gitignore

This had been an issue for some of the users. The mangle between filesystem, permissions and symlinks had been a source of problems.
This PR removes the need for symlinking because the assets will be cached in the /public folder. Since the public folder is not .gitignored by default, the install command now adds the basset folder to the gitignore.

Cache URL's in dev mode

One painpoint of using basset in development was that you couldn't cache the http dependencies (usually cdns) and keep your "local" files "uncached", so that you could work developing your admin panel without internet access.
In this PR we introduce the force_url_cache, that will cache the urls even when you are in dev mode. That way it's possible to cache everything in advance with php artisan basset:cache, or at the required time when the asset is saw on the page for the first time.

Local files (recaching issues)

While working locally, all local assets (eg, vendor/backpack/crud/resources/assets/css/common.css) were "re-cached" on every request. That was required to make the assets available "for public" usage (the vendor folder is not accessible).
In this PR we introduced an "hash" that we store alongside the local files in the file map. That way, we only need to "re-cache", the ones were the content changed.

Named Assets

This is the block muscle of this PR. By using named assets we unlocked a lot of other possibilities.
The concept is simple, packages (like backpack/crud), register their named assets with Basset::map(), when refering to those packages, developers can use only @asset('asset-package-name').
The advantages of this approach are:

  • packages that use the same asset in multiple places (eg: select2 in 10+ files), just need to update the Basset::map() definition in 1 place, ensuring that no package conflicts occur by using multiple versions of the same package.
  • the usage of "variables" in assets was not possible. With named assets, developers can register in advance the "possible" variables, so that no "on-the-fly" caching happens.
  • When a package version change, we can delete the previous version, and get the new one.

All this now enables the workflow of publishing your assets to your repo, removing the need for calling php artisan basset:cache in production. Or by using a combination of both, and just calling :cache without :clear, so only the "changed" assets will be cached. If nothing changed, no assets are "re-cached".

@pxpm pxpm added the v7 label Dec 4, 2024
@tabacitu
Copy link
Member

tabacitu commented Dec 9, 2024

OMFG I absolutely LOVE the description here @pxpm . I LOVE IT! Brief but comprehensive, and crystal-clear. Let's use this as the changelog for this version - it's EXCELLENT.

Copy link
Member

@tabacitu tabacitu left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hey @pxpm - as much as I loved your description for this PR... the diff is difficult to read. man.

I've spent 2h reading this PR and still don't feel like I understand it at all.

Seems like EVERYTHING changed, everything became more complicated, and I'm struggling to understand the reasons for the changes, to be able to judge the decisions you've made. I'm sure it works, but I can't really accept this PR until I understand it. Until I know we get it to be cleaner than what we had before. And that will take us few back-and-forths, from what it looks.

  1. First up, please take some time to reply to my answers here... everything is surface-level because (again) I don't understand it.

  2. Second, please take some time to describe the changes to me, from a technical perspective. Broad picture. Eg.

  • an asset used to be a string, but now it's an object and I called that object cacheEntry;
  • our cachemap used to X, now it Y;
  • this is new to me too... but there are just too many changes here, in every file... for me to understand WHY they have all changed; maybe a summary is possible, and that would help me... I don't know;
  1. If you have time, please also do a self-review. Erase everything from your memory and review this DIFF as if it were your first time trying to understand what happened here. See if you'd change anything for readability, clarity, brevity, etc.

We can do this 💪

src/AssetOverwrite.php Outdated Show resolved Hide resolved
src/config/backpack/basset.php Outdated Show resolved Hide resolved
src/config/backpack/basset.php Outdated Show resolved Hide resolved
src/BassetServiceProvider.php Show resolved Hide resolved
src/Console/Commands/BassetCache.php Outdated Show resolved Hide resolved
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is a very nice class from a programming perspective. The names seem good as well. Please add comments and explanations, so that it's easy for a human to read it as well. It's got 18 methods so feel free to split them up as well, to make it clear what the affordances are for this class.

src/Helpers/FileOutput.php Outdated Show resolved Hide resolved
src/Helpers/FileOutput.php Outdated Show resolved Hide resolved
Comment on lines 56 to 62
if (! file_exists(base_path('.gitignore'))) {
$this->components->task($message, function () {
file_put_contents(base_path('.gitignore'), 'public/'.config('backpack.basset.path'));
});

return;
}
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Wait. Why do we add the basset path to .gitignore on installation? Wasn't the whole point of publishing to the public directory... to have the assets in source control?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hey Cristian, my understanding from last iteration we had is that we would keep the "current behavior" as the default, but removing from .gitignore and pushing the public assets to the repo would work as if they were regular files, namely only "changed files" will appear in the diff.

We can change the default, but for that I would argue that we need a way to "don't load" some assets (things you are not using). I thought about Basset::map('asset-name', null) to don't load that specific asset, but it seems cumbersome, so maybe Basset::dontLoad(['asset-1', 'asset-2']) would be better (defined in the same OverrideAssets class).

But there is also another issue, if the basset isn't "named", so a @bassetBlock or something similar we would always load it, and I think that's not the desired behavior too. eg:
4 wysiwyg editors. they all load the "editor script" and the "editor initialization code". You only use 1 of them, so you could easily do Basset::dontLoad(['editor-1', 'editor-2', 'editor-3']). That wouldn't load those editors from cdns, but the "initialization scripts" for those editors would still be loaded. Similar issue with the CSS.

I think one reasonable solution would be to have "grouped assets", so that the editor.js, editor.css and the @bassetBlock('editor-initialization') could be disable in one go by disabling the whole group. This will work if we have "related code blocks/urls" in different files.

If we wanted to simplify things as everything is usually on the same file, we could add excluded_views that would not cache anything inside those files.

I think this is kind of a must when you are publishing assets to your repo, you wouldn't want files that you are not using, either for consistency or for saving space/computation time.

I didn't implemented nothing like that yet, maybe we can talk a bit about it and decide if it's worth to do right now and how to do it. 👍

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

After further review, I agree with your decision here - .gitignore will work best in 99% of the cases, so we keep it as the default, yes.

For the assets-committed-to-git use case... I understand why a new function would be needed, yes. I don't think it's urgent to create one, but here are my thoughts:

Basset::map('asset-name', null); 
// I like it, it simple and clean;

Basset::dontLoad(['editor-1', 'editor-2', 'editor-3']); 
// I don't like the "dont", and don't like the "load" - does that mean it doesn't cache it or what? not explicit enough;

// maybe we can do something like
Basset::ignore('asset-name');
// but this still doesn't answer the whole question... what will happen... 
// - will it not get cached on disk?
// - will it not get loaded onpage at all? or will it be loaded from the CDN?

Regarding grouping, I would NOT add grouping. That will add a lot of complexity to our code, for a 1% scenario. It would probably be easier to just add something like this, to allow vendors to create namespaces for names:

Basset::ignore('ckeditor.*'); 

But even that... to be honest, would add complexity, and I don't know if it's worth it at this point.

--

If you think we should do this... please add an issue for it with our best thoughts... but personally I don't think this is URGENT nor IMPORTANT so I wouldn't spend time on this right now.

Comment on lines 486 to 491
// when in dev mode, cdn should be rendered
if ($this->dev && ! $this->forceUrlCache) {
$output && $this->output->write($asset, $this->dev);

return $this->loader->finish(StatusEnum::DISABLED);
}
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I see too many places in the code where dev_mode is mentioned... and I don't understand how it works any more. Previously, dev_mode meant assets weren't actually internalized. Now... what does it mean?

Please explain to me:

  • what dev_mode does in this current version;
  • why you should use it;
  • why you would need to separately set dev_mode and force_url_cache;

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Dev mode has 3 tasks:
1 - do not cache external asset urls
2 - do not cache files located in accessible folders (eg: /public/my-assets/asset.js)
2 - refresh the "forced" local cached files. (files that are located in the vendor/ folder that need to be moved to accessible folders before used).

Using the force_url_cache will just change the behavior of the 1 task.

In addition to this change, dev mode will now also compute and compare the basset file hashes of the vendor files and only perform the internalization of the ones that changed.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Sorry but just tried it and... I don't see that behaviour. And I still don't understand what dev_mode does now. This is too convoluted. Here's what I've done:

--

I've run php artisan basset:clear, have no basset entries in the .ENV file and here's my config file:

    // development mode, assets will not be internalized
    'dev_mode' => env('BASSET_DEV_MODE', env('APP_ENV') === 'local'),

    // all external urls (usually cdns) will be cached on first request or
    // when running `basset:cache` even if dev_mode is enabled
    'always_cache_external_urls' => false,

Due to the comments in this config file (and your previous comment in this thread), I expect that no assets will get internalized. That it will load from the CDNs directly. Because my APP_ENV=local and that's what the comment above dev_mode says "assets will not be internalized".

I open the page and... after I hard-refresh... I get a broken page:

CleanShot 2024-12-20 at 10 07 15@2x

and some assets DID get internalized (the vendor ones from what I can tell):

CleanShot 2024-12-20 at 10 07 37@2x

If I click "view page source" I see that the URLs the page tries to load are the "internalized URLs", but those files aren't there.


This was NOT my expectation. The change in behaviour of dev_mode doesn't make sense to me - I literally don't know how to use it. This new always_cache_external_urls config also doesn't make sense to me, especially in combination with dev_mode. It's an always... so if I turn it "false" is becomes... not-always... what does "not always" mean... in what cases is it enabled?! 😡

To me, the previous behaviour of dev_mode was easy to understand - you flip that switch, and the Basset will do nothing except the absolute mandatory. Now... I don't know what dev_mode does. And there's a new config that says it will basically ignore the previous config?! I can't wrap my head around it. And if I can't, nobody will.


Since your explanation still didn't clarify to me how I can use dev_mode... or why I should... it might be smart for me to wait for you to write the README. Maybe after you do that... I will understand what dev_mode does and when I should use it.

But if it's not a me-being-stupid problem, and more of a this-being-way-to-convoluted problem.... I might have a way out, a way to clarify what dev_mode does AND to customize it. Here goes. What if we:

  • keep the BASSET_DEV_MODE entry in the .ENV file as the toggle;
  • remove the dev_mode entry from the config file
  • add config entries for every little thing DEV_MODE disables, and use the BASSET_DEV_MODE env as the default for them;

I think pseudocode will explain it better. Something like this:

    // this entry goes away entirely
    // 'dev_mode' => env('BASSET_DEV_MODE', env('APP_ENV') === 'local'),

    // do not cache assets from external urls
    'do_not_cache_external_urls' => env('BASSET_DEV_MODE') ?? false,

    // do not cache files located in accessible folders (eg: /public/my-assets/asset.js)
    'do_not_cache_files_in_accessible_folders' => env('BASSET_DEV_MODE') ?? false,

    // do not trust cached code blocks (eg. loaded with @bassetBlock)
    // instead of loading the cached code block, should check if it's changed and re-cache if needed
    'do_not_trust_cached_code_blocks' => env('BASSET_DEV_MODE') ?? false,

    // do not trust cached local files, eg. files that are located in the vendor/ folder and were moved 
    // to accessible folders before used; should check if changed and re-cache if needed
    'do_not_trust_cached_local_files' => env('BASSET_DEV_MODE') ?? false,

Wouldn't this clarify EXACTLY what dev_mode does? And give you granular control over it?

If we are to do this, please do it as a separate PR, I'm not 100% sure we should do it either.

@tabacitu tabacitu assigned pxpm and unassigned tabacitu Dec 9, 2024
@tabacitu tabacitu changed the title [v2] - New basset version [WIP] Next basset version Dec 16, 2024
Copy link
Member

@tabacitu tabacitu left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hey Pedro! I love the PR in the end 😀 Took me a while to wrestle with it... In the future let's not leave so much work in one PR before we review... we're just making it too difficult to ourselves.

After using it and going through all the code, I think I understand it now - thank you for your patience. I noted here quite a few suggestions (most minor / DX).

In addition, please:

  • add a new section to the README file, an UPGRADE GUIDE;
  • go through the rest of the README file, to make sure it details the new behaviour (not the old one) and all of the things people can do with it;

src/BassetServiceProvider.php Show resolved Hide resolved
src/BassetServiceProvider.php Show resolved Hide resolved
* @return void
*/
private function createSymLink(): void
private function addGitIgnore()
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What do you think about making this optional? To ASK the user whether they want to do this, upon installation? That's a great moment to give them the choice, right?

Eg.:

Would you like to add the 'public/basset' directory to .gitignore? This means cached assets will not show up in your git history, if you use the public_basset disk. (Yes/No; default: Yes) More info: https://bla-bla

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Then that could link to the section in the docs where we explain the benefits of each option:

  • if you add this to .gitignore, you should have php artisan basset:cache to your deploy script;
  • if you do NOT add it to .gitignore, you are 100% sure you have the same assets in local and staging, but your git history will also have the changes to your CSS and JS dependencies;

src/Console/Commands/BassetCache.php Outdated Show resolved Hide resolved
@@ -20,8 +20,8 @@ class BassetServiceProvider extends ServiceProvider
Console\Commands\BassetClear::class,
Console\Commands\BassetCheck::class,
Console\Commands\BassetInstall::class,
Console\Commands\BassetInternalize::class,
Console\Commands\BassetFresh::class,
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@pxpm can you please check that this command does what we expect it to do? Last I remember, it didn't.

I really want us to be able to tell people they can safely run php artisan basset:fresh - it's especially useful in the scenario where you do NOT have the files in gitignore. You don't only want to add new files, you also want to delete the old ones, right?

Copy link
Member

@tabacitu tabacitu Dec 18, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Not directly to this PR, but here's something I discovered:

  • the new ckeditor addon does NOT add its path to Bassets; so its asset aren't cached when you run php artisan basset:cache; I assume it's the same for the new tinymce addon;
  • the CSS and JS we use in our Chart Widget (in PRO) is not internalized upon php artisan basset:cache
  • FileManager does NOT add its path to Bassets, or at least when you load it, it caches more files;

These are the few cases I found we can probably fix. But my point here is this: I don't think it's SAFE for anybody to assume that php artisan basset:cache will actually cache everything you are using. I think we should heavily recommend that people use the .gitignore, unless they know what they're doing and want full control.

Why? Because if you are not using .gitignore for that directory... and you missed just ONE file on localhost that you didn't cache... the downside is big. Your deployment script stops working. So I am 90% sure people who go down this route... will end up with conflicts in production.

Let's fix the problems above in their respective directories (or leave this open until we do). But I wanted to take the opportunity to detail why I believe .gitignore should be our default and recommended option.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

That should be fixed in Laravel-Backpack/CRUD#5720 , https://github.com/Laravel-Backpack/PRO/pull/292 , Laravel-Backpack/FileManager#64 , Laravel-Backpack/theme-tabler#194 & other themes.

I may have missed some of them in other packages so we should carefully check them.

Theoretically we must ensure that a developer can do a basset:cache and all "Backpack" assets get cached, but for the reason you mentioned, I opted to have the .gitignore by default until some people conscientiously change it to have the assets in the repository and may find anything we've missed.

The aim, like I said, should be that EVERYTHING Backpack related can be cached manually.

src/Helpers/CacheEntry.php Show resolved Hide resolved
src/Helpers/CacheEntry.php Outdated Show resolved Hide resolved
src/Helpers/CacheEntry.php Outdated Show resolved Hide resolved
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This command... didn't work for me, it didn't output anything:

CleanShot 2024-12-18 at 14 00 03@2x

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The command totally works, but that list only the named assets, things that you can directly copy to your OverrideAssets class and change them. The other can't be changed without publishing some view and changing it.

For that to work you must be on the branches I mentioned in other comment: #143 (comment)

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Oohhh ok. Then let's have it output something (anything) in this use case. Maybe smth like:

  • "No named assets have been found in the cachemap"
  • "End of named assets list"
  • "0/300 assets in the cachemap have names"

Whatever is easiest, but an output nonetheless.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Projects
Status: In Review
Development

Successfully merging this pull request may close these issues.

2 participants