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

feat: Adds Plugins feature #2563

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

Conversation

ArthurTriis1
Copy link
Contributor

@ArthurTriis1 ArthurTriis1 commented Nov 14, 2024

What's the purpose of this pull request?

Tip

PT-BR: Para melhor compreensão da feature recomenda-se assisti a apresentação feita ao time da Faststore Experience, explicando as motivações da feature e uma demo do seu funcionamento.

Note

This PR might seem extensive, but the majority of the changes are focused on the hCMS.ts and plugins.ts files. The other files are mostly support files for the features and will have their content overwritten during the generate process.

This PR adds the plugin feature to Faststore.

The Faststore Plugins feature enables the addition of new functionalities to a store without requiring direct integration into the Faststore core or the store's source code. This approach not only maintains source code encapsulation but also extends the extensibility options already available in Faststore, such as Component Override, theming, and CMS customization, while introducing new capabilities.

Plugins operate as an intermediary layer between the Faststore core and the store code. This means that modifications made by a plugin override core functionality but can still be overridden by the store's customizations, ensuring flexibility and adaptability.

One notable enhancement brought by this feature is the ability to create new pages, something previously unsupported directly within the store. For instance, the Buyer Portal — a B2B solution — requires creating a management page and modifying existing Faststore components. With Plugins, such functionality can be implemented in a modular and scalable way, empowering developers to build solutions without impacting the core or existing store customizations.

The plugins feature is in the early stages of development and and plugins can only be developed by VTEX.

Screenshot 2024-11-14 at 17 54 49

Implemented Features

  • New Pages: Plugins enable the creation of new pages in a store.
  • Plugin Override Components: New sections can be created via plugins. These sections override core sections but can, in turn, be overridden by store-specific sections.
  • hCMS Merge: The content-types.json and sections.json files are merged across core, plugin, and store files, with the store files taking precedence.
  • Theme Merge: Plugins can add theme overrides and customizations. These plugin-level theme changes can be further overridden by store-specific themes.

How it works?

Configuring a New Plugin

To add custom functionalities to your Faststore store, you can create a new plugin. This plugin can include custom pages, UI components, themes, and CMS configurations, all in a modular way.

Plugin Package Structure

The plugin should follow this folder structure:

@faststore/new-plugin/
├── src/                        
│   ├── pages/                  # Custom pages
│   ├── components/             # UI components
│   │   └── index.tsx            # Example UI component
│   ├── theme/                  # Custom themes and styles
│   │   └── index.scss          # Styles file
├── cms/                        
│   ├── faststore/              
│   │   ├── content-types.json  # CMS content types
│   │   └── sections.json       # Custom CMS sections
├── package.json                # Plugin metadata and dependencies
├── yarn.lock                   # Yarn dependencies and versions
├── plugin.config.js            # Plugin configuration
├── tsconfig.json               # TypeScript configuration
  • src/: Contains the plugin source code, such as custom pages (pages/), UI components (components/index.tsx), and themes (theme/index.scss). All content within src/ will be copied to the plugins/ folder inside .faststore when the plugin is integrated into the store.

  • cms/: Contains configuration files for CMS integration, as content-types.json and sections.json.

  • plugin.config.js: The plugin configuration file where page paths are defined.

Dependencies

You can add dependencies to your plugin, such as Faststore packages to access core and UI functionalities, as well as specific packages for the framework, such as Next.js 13, if required. These dependencies should be added to the plugin's package.json, enabling advanced features within the plugin code.

Creating a New Page

To create a new page in your plugin, you need to define its configuration in the plugin.config.js file. The configuration structure should look similar to the example below:

module.exports = {
  name: "poc-plugin",
  pages: {
    "my-account": {
      path: "/my-account",
      appLayout: false,
    },
  },
};

Key Properties:

  • Page Name: In this example, "my-account" is the name of the page, and this should match the name of the file in the pages/ folder (e.g., my-account.tsx).
@faststore/new-plugin/
├── src/                        
│   ├── pages/                    # Custom pages
│   │   └── my-account.ts   # Example page
  • path: This defines the route where the page will be registered. It follows the same pattern as the Next.js page router, meaning the page will be accessible via the defined path (e.g., /my-account).
  • appLayout: This property defines whether the global components (like the header and footer) will be displayed on the page. Set it to false to disable the default layout.

Page Structure

Each page should have at least the following structure. The loader function will be executed server-side:

// loader function to fetch data
export async function loader() {
  const result = await fetch("http://...");

  return await result.json();
}

// Page component to render data
export default function MyAccount(data: any) {
  return (
    <></>
  );
}

Creating New Sections and Overriding Existing Sections

To create new sections or override existing ones, you should follow a structure similar to the one used in Faststore stores. The sections should be placed inside the /components folder and the index.tsx file must export all your sections as an object, with each section as a property of that object

Section Structure

The new sections will be made available, and any sections with the same name as the core sections will override those sections.

@faststore/new-plugin/
├── src/                        
│   ├── components/             # UI components
│   │   └── ProductDetails.tsx            # Example override section
│   │   └── ProductInfo.tsx            # Example new section
│   │   └── index.tsx            # File to exports components

Here's an example of how to define sections in an index.tsx file:

import PersonalInfo from "./PersonalInfo";
import ProductDetails from "./ProductDetails";

const sections = {
  ProductDetails,
  PersonalInfo,
};

export default sections;

In this example:

  • PersonalInfo: is a new section that has been created.
  • ProductDetails: is an override of the default component from Faststore.

Customizing CMS via Plugin

To customize the CMS via a plugin, you need to create two files: sections.json and content-types.json. These files should be placed under the /cms/faststore/ folder in your plugin.

File Structure:

@faststore/new-plugin/
├── cms/
│   ├── faststore/
│   │   ├── sections.json
│   │   └── content-types.json
  • sections.json: Defines the sections available in the plugin.
  • content-types.json: Defines the content types handled by the plugin.

Merging Files

To merge your plugin’s CMS files with the store’s files, use the cms-sync command. This will merge the plugin’s sections.json and content-types.json with the store’s, giving priority to the store's content.

cms-sync

This command ensures that:

  • New sections and content types from the plugin are added.
  • Sections/content types with the same name as the core will override the default ones.
  • The store's customizations overrides all.

Creating a Theme

To create a theme for your plugin, the process is similar to creating a theme for a store. The theme should be defined in the src/themes/index.scss file, where all the CSS for the plugin will be written.

File Structure:

@faststore/new-plugin/
├── src/
│   ├── themes/
│   │   └── index.scss

index.scss: This is where all the styles for the plugin should be defined.

CSS Loading Order

The CSS from the plugin's index.scss will be loaded after the global styles and before the store's CSS, allowing it to override the store's styles while respecting the global styles.

Adding a Plugin to the Store

To add a plugin to your store, you need to follow these steps:

1. Install the Plugin Package

First, install the plugin package in the store’s codebase. For example, if you are adding the @faststore/plugin-test, run:

yarn add @faststore/plugin-test

2. Update discovery.config.js

In the discovery.config.js file, add the plugin name to the plugins property. This will register the plugin for use in the store.

Copy code
module.exports = {
  ...
  plugins: [
    "@faststore/plugin-test",
  ],
  ...
}

3. Configure Plugin Page Paths (Optional)

You can also configure the path for the plugin's pages in the discovery.config.js file. For example, to change the default path for the my-account page, use the following structure:

module.exports = {
  ...
  plugins: [
    {
      "@faststore/plugin-test": {
        pages: { "my-account": { path: "/other-path" } },
      },
    },
  ],
  ...
}

This allows you to specify custom paths for any pages defined by the plugin.

How to test it?

Testing Locally

To test your plugin locally, you need to link the core package, CLI, and the plugin to a starter store. Here's how you can do it:

1. Link the Core, CLI, and Plugin Packages

First, you will need to link the core, CLI, and the plugin to your local starter store. You can do this using yarn link for each package to the starter from this PR.

Link the core and CLI from the current PR
Link the plugin from this repository.

2. Run the Starter Store

Once linked, navigate to the starter store's directory and run the commands as you normally would in a store:

yarn install
yarn dev
  1. Server Restart for Changes
    Every time you make a change to the plugin, you will need to restart the server for the changes to take effect.

For changes made to the core and CLI, ensure they are rebuilt before testing:

yarn build

Starters Deploy Preview

Testing on Preview

To test your plugin on the preview environment, follow these steps:

1. Preview the Starter Store with Plugins

There is an open PR that allows you to preview the store with plugins running. You can access the preview environment using the following link:

Preview Store

This preview is based on the starter store from the PR starter.store/pull/616.

2. Check the Main Color Override

On the homepage, you will notice that the main color has been overridden by red, which is done via the plugin. This demonstrates that the plugin is successfully applying styles to the store.

Screenshot 2024-11-19 at 10 56 19

3. Test the my-account Page

You can access the my-account page by navigating to /my-account in the preview store. On this page, you will see the result of the useSession hook displayed in blue.
My Account

Screenshot 2024-11-19 at 10 56 51

4. Test the Product Detail Page (PDP)

On the Product Detail Page (PDP), you will see the overridden component from the plugin. This component displays:

The result of the useSession hook.
A Next.js link to navigate back to the homepage.
You can test this by visiting a PDP, such as:

Product Detail Page

Screenshot 2024-11-19 at 10 57 46

References

B2BTEAM-1951
starter.store/pull/616.
Preview Store
POC-Plugin

Checklist

You may erase this after checking them all 😉

PR Title and Commit Messages

  • PR title and commit messages follow the Conventional Commits specification
    • Available prefixes: feat, fix, chore, docs, style, refactor and test

PR Description

  • Added a label according to the PR goal - breaking change, bug, contributing, performance, documentation..

Dependencies

  • Committed the yarn.lock file when there were changes to the packages

Copy link

vercel bot commented Nov 14, 2024

The latest updates on your projects. Learn more about Vercel for Git ↗︎

1 Skipped Deployment
Name Status Preview Comments Updated (UTC)
faststore-site ⬜️ Ignored (Inspect) Visit Preview Jan 13, 2025 2:20pm

Copy link

codesandbox-ci bot commented Nov 14, 2024

This pull request is automatically built and testable in CodeSandbox.

To see build info of the built libraries, click here or the icon next to each commit SHA.

@ArthurTriis1 ArthurTriis1 force-pushed the feat/add-plugins-feature branch from 39c2a61 to bb8f5c0 Compare November 14, 2024 18:32
@ArthurTriis1 ArthurTriis1 force-pushed the feat/add-plugins-feature branch from cff672f to 61a18b3 Compare November 14, 2024 20:21
@ArthurTriis1 ArthurTriis1 changed the title Feat/add plugins feature feat Adds Plugins feature Nov 14, 2024
@ArthurTriis1 ArthurTriis1 added the enhancement New feature or request label Nov 14, 2024
@ArthurTriis1 ArthurTriis1 self-assigned this Nov 14, 2024
@ArthurTriis1 ArthurTriis1 marked this pull request as ready for review November 14, 2024 23:16
@ArthurTriis1 ArthurTriis1 requested a review from a team as a code owner November 14, 2024 23:16
@ArthurTriis1 ArthurTriis1 requested review from renatamottam, lucasfp13, gvc and eduardoformiga and removed request for a team November 14, 2024 23:16
@hellofanny hellofanny added the contributing Pull request submitted by the community label Nov 19, 2024
Copy link
Contributor

@gvc gvc left a comment

Choose a reason for hiding this comment

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

Added a couple of questions :)

* If it reaches process.cwd() (or /, as a safeguard), without finding it, it will throw an exception
*/
const getCorePackagePath = () => {
const coreFromNodeModules = path.join('node_modules', '@faststore', 'core')
const packageFromNodeModules = path.join(
Copy link
Contributor

Choose a reason for hiding this comment

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

Why was this variable renamed? If I understood correctly, most changes on this file are cosmetic (indentation) and you're adding the getPackagePath entry to the returned object, right? If that's so can you undo the rename of this const?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Good, it must have been a mistake. Renamed.

packages/cli/src/utils/hcms.ts Show resolved Hide resolved
@@ -1,9 +1,15 @@
import WebFontsOverrides from 'src/customizations/src/components/overrides/WebFonts'
import { default as CoreWebFonts } from 'src/fonts/WebFonts'
import ThirdPartyScriptsOverrides from 'src/customizations/src/components/overrides/ThirdPartyScripts'
import ThirdPartyScriptsPluginsOverrides from 'src/plugins/overrides/ThirdPartyScripts'
Copy link
Contributor

Choose a reason for hiding this comment

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

Does B2B have any specific needs for Third Party Scripts or Web Fonts? Or has this just been added for consistency/completeness? If it's just for completeness, I'd remove it.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

We need the WebFonts, but Third-Party Scripts are not required yet, so they have been removed.

packages/core/src/plugins/index.scss Show resolved Hide resolved
@ArthurTriis1 ArthurTriis1 requested a review from gvc January 2, 2025 22:16
Copy link
Contributor

@gvc gvc left a comment

Choose a reason for hiding this comment

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

Small issues but look good overall. I'd only do extra testing for breaking changes on the CLI.

packages/cli/src/utils/hcms.ts Show resolved Hide resolved
packages/core/src/plugins/index.scss Show resolved Hide resolved
packages/core/src/plugins/index.scss Show resolved Hide resolved
Copy link
Contributor

@gvc gvc left a comment

Choose a reason for hiding this comment

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

When running this build of the CLI with a store without a plugins key on faststore.config.js, I got this error: TypeError: Cannot read properties of undefined (reading 'forEach') Which means that unless we can handle a faststore.config.js without a plugin entry, this will have to be a breaking change. Can you just handle the case of it not having the plugin entry?

@ArthurTriis1 ArthurTriis1 force-pushed the feat/add-plugins-feature branch from d526b1c to 4aaedb3 Compare January 13, 2025 13:16
@ArthurTriis1 ArthurTriis1 changed the title feat Adds Plugins feature feat: Adds Plugins feature Jan 13, 2025
@ArthurTriis1
Copy link
Contributor Author

When running this build of the CLI with a store without a plugins key on faststore.config.js, I got this error: TypeError: Cannot read properties of undefined (reading 'forEach') Which means that unless we can handle a faststore.config.js without a plugin entry, this will have to be a breaking change. Can you just handle the case of it not having the plugin entry?

Yes! I fixed it in this comment 4aaedb3

@ArthurTriis1 ArthurTriis1 force-pushed the feat/add-plugins-feature branch from e5e4041 to a0fbd79 Compare January 13, 2025 14:20
@ArthurTriis1 ArthurTriis1 requested a review from gvc January 13, 2025 15:11
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
contributing Pull request submitted by the community enhancement New feature or request
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants