c975l/shop-bundle

Shop Bundle for eCommerce with Symfony

v0.20 2025-04-05 17:09 UTC

README

Shop Bundle for eCommerce with Symfony

BUNDLE UNDER DEVELOPMENT, USE AT YOUR OWN RISKS

With is bundle, you'll be able to manage a shop + crowdfunding.

## Installation

First, launch composer require c975l/shop-bundle to install the bundle.

Create database tables : php bin/console make:migration and php bin/console doctrine:migrations:migrate.

Crreate a private at your root level and add it to your .gitignore file.

Configuration

The bundle relies on the use of App\Entity\User. If you haven't, create it with php bin/console make:user. Then add one and give it ROLE_ADMIN. You can use php bin/console security:hash-password to hash the password.

Create a login form/logout route: php bin/console make:security:form-login. Then adapt `` to your needs.

Add the following configuration in the different files:

# config/packages/security.yaml
security:
    firewalls:
        main:
            logout:
                path: app_logout
    access_control:
        - { path: ^/shop/management, roles: ROLE_ADMIN }
# config/packages/vich_uploader.yaml
vich_uploader:
    db_driver: orm

    metadata:
        type: attribute

    mappings:
        products:
            uri_prefix: '' # path added in Listener
            upload_destination: '%kernel.project_dir%/public/medias/shop/products'
            namer: Vich\UploaderBundle\Naming\UniqidNamer
            inject_on_load: false
            delete_on_update: true
            delete_on_remove: true
        productItems:
            uri_prefix: '' # path added in Listener
            upload_destination: '%kernel.project_dir%/public/medias/shop/items'
            namer: Vich\UploaderBundle\Naming\UniqidNamer
            inject_on_load: false
            delete_on_update: true
            delete_on_remove: true
        productItemsFiles:
            uri_prefix: '' # path added in Listener
            upload_destination: '%kernel.project_dir%/private/medias/shop/items' # Has to be outside of public folder, otherwise accessible to anyone, and added in .gitignore
            namer: Vich\UploaderBundle\Naming\UniqidNamer
            inject_on_load: false
            delete_on_update: true
            delete_on_remove: true

        crowdfundings:
            uri_prefix: '' # path added in Listener
            upload_destination: '%kernel.project_dir%/public/medias/shop/crowdfundings'
            namer: Vich\UploaderBundle\Naming\UniqidNamer
            inject_on_load: false
            delete_on_update: true
            delete_on_remove: true
        crowdfundingsCounterparts:
            uri_prefix: '' # path added in Listener
            upload_destination: '%kernel.project_dir%/public/medias/shop/counterparts'
            namer: Vich\UploaderBundle\Naming\UniqidNamer
            inject_on_load: false
            delete_on_update: true
            delete_on_remove: true
            delete_on_remove: true
# config/routes.yaml
c975_l_shop:
    resource: "@c975LShopBundle/"
    type:     attribute
    prefix:   /

Configure the webhook in Stripe dashboard

Create a Stripe account if not: https://stripe.com

  1. Sign in to your Stripe Dashboard
  2. Navigate to Developers > Webhooks
  3. Click "Add endpoint"
  4. Enter your webhook URL (https://your-website.com/shop/stripe/webhook)
  5. Select the event checkout.session.completed
  6. Copy the webhook signing secret and add it to your environment variables config/config_bundles.yaml -> stripeWebhookSecret
  7. Test the endpoint to ensure proper configuration

This webhook allows Stripe to notify your application when payments are completed, ensuring order processing even if customers close their browser after payment.

Create the configuration file config/config_bundles.yaml with these settings:

c975LShop:
    name: 'My Shop' # Name of the shop
    roleNeeded: 'ROLE_ADMIN' # Role needed to access shop management
    from: 'contactp@example.com' # Email address for sending emails
    fromName: 'My Shop' # Sender name
    replyTo: 'contact@example.com' # Reply-to email address
    replyToName: 'My Shop' # Reply-to name
    currency: 'eur' # ISO currency code
    shipping: 500 # Shipping cost in cents (5.00)
    shippingFree: 10000 # Free shipping threshold (100.00)
    sitemapBaseUrl: 'https://example.com'  # Base URL for sitemap
    stripeSecret: 'STRIPE_SECRET' # Stripe secret key
    stripeWebhookSecret: 'STRIPE_WEBHOOK_SECRET' # Stripe webhook secret
    touUrl: 'https://example.com/terms-of-use' # Terms of use URL
    tosUrl: 'https://example.com/terms-of-sales' # Terms of sales URL

In assets/bootstrap.js, add the following code:

import c975lShopBasket from '/bundles/c975lshop/js/basket.js';

app.register('basket', c975lShopBasket);

Launch configuration process php bin/console config:create.

CSP

Adapts your CSP to allow:

script-src: unsafe-inline

Urls

You should be able to use your shop + Management with the following urls:

Useful Commands

The basket.digital has 3 values: 1 (digital), 2 (both) and 3 (physical).

The basket.status are the following: new, validated, paid, downloaded, shipped, finished

Run this Command php bin/console shop:downloads:delete once a day to delete files made available at download.

Run this Command php bin/console shop:products:position once a day to correct position and keep a 5 gap between them.

Run this Command php bin/console shop:baskets:delete once a day to delete unvalidated baskets (status new and creation > 14 days).

For creating the sitemap, you can run php bin/console shop:sitemaps:create thath will give a public/sitemap-shop.xml that you can add to your sitemap-index.xml file or run the following:

    //Creates the sitemap for pages managed by Shop
    public function createSitemapShop($output)
    {
        $command = $this->getApplication()->find('shop:sitemaps:create');
        $inputArray = new ArrayInput([]);
        $command->run($inputArray, $output);
    }

Entities structure and nesting

  • Product [Collection]
    • ProductMedia [Collection]
    • ProductItem [Collection]
      • ProductItemMedia [One]
      • ProductItemFile [One]

Shop Bundle Entity Structure

TODO

In src/Listener/ProductItemListener.php we need to create an empty ProductItemMedia|ProductItemFile if none is added, otherwise we can't add one afterwards. The physical ProdutItemMedia is not deleted when the ProductItem is deleted, but the link is removed. See ProductItemListener->prePersist(). Furthermoe, need to create ProductItem without Meida/File first.

A Command has been made to remove those files, simply run (and/or add incrontab) php bin/console shop:media:delete.