Version Deployment
There are two main reasons why Freemius manages its own repository and deployment process:
- To be able to securely serve paid plugin/theme downloads and updates to customers who own a valid license.
- To simplify code management by allowing developers to maintain only one code base which contains all of their plugin/theme functionality. A free, WordPress.org compliant version is automatically generated for them, using a custom PHP Preprocessor.
Deploying to Freemius
Manual Deployment
- Login to your Freemius Dashboard and head over to the Deployment section.
- Zip your plugin's or theme’s root folder and upload it by clicking the [ + Add New Version] button, located at the top.
Deployment Automation
deploy-on-freemius: a GitHub action for deployment and releasing a new product version via Freemius.
Credits: Created & maintained by Jasper Vriends / Buttonizer
freemius-suite: a Python based Bash library for packaging, deployment, and releasing a new product version via Freemius.
Credits: Created & maintained by Daniele Scasciafratte
gulp-freemius-deploy: a niffty Gulp NPM library for packaging and deploying a new product version via Freemius.
Credits: Created & maintained by James Kemp
Freemius API: a PHP example of how to deploy programatically using Freemius REST API.
What Happens When You Deploy: Free + Paid Packages
The processor will auto-generate two versions of your plugin/theme:
- A paid version: One that's identical to your uploaded version, which includes all of the code. Will be available for download ONLY for customers with a valid license (paid or trial).
- A free version: The same code, stripped from all of your paid features. This stripped down version is what your users will be able to download first. In case your original plugin/theme is compatible with the official WordPress.org guidelines - the free version, will also be compatible, as all paid code was stripped.
The paid version will not be available for download or update for your customers until you change the release status to Released. If you have a beta program, you can also flag the release as Beta release to make it only available for customers who are part of the beta program.
If you already developed free and paid versions that are fundamentally different, you can either merge the versions into a single code base or ignore the free version generated by Freemius. If you go with the 2nd option, make sure also to integrate the SDK into your free version and set the is_premium flag to false to indicate the SDK that it is running in the scope of the free version.
Free Version Auto Generation / Paid Logic Stripping / PHP Preprocessor
The deployment PHP Preprocessor will automatically strip out any paid code - based on the following rules:
Stripping Paid Only PHP Logic
All logic within the if statements containing usage of license related methods with the __premium_only suffix will automatically be stripped from the free version.
// This IF block will be auto removed from the Free version.
if ( my_fs()->is__premium_only() ) {
    // This IF will be executed only if the user is in trial mode or has a valid license.
    if ( my_fs()->can_use_premium_code() ) {
        // ... premium only logic ...
    }
}
 
// This "if" block will be auto removed from the Free version.
if ( my_fs()->is_plan__premium_only('starter') ) {
    // ... logic related to starter plan and higher plans ...
}
Stripping Paid Only CSS & JavaScript Logic
Simply wrap the code you'd like to exclude with the following meta comments:
(function($){
    /*! <fs_premium_only> */
    // ... my premium only code ...
    /*! </fs_premium_only> */
})(jQuery);
In the future, we'll provide a richer JavaScript SDK which will handle license-related logic in a similar manner to how our PHP SDK does it.
Developing a Gutenberg or JavaScript driven plugin with webpack?
To preserve Freemius meta comments from being stripped by webpack's uglifyjs, configure webpack.config.js as follows:
module.exports = {
  optimization: {
    minimizer: [
      new UglifyJsPlugin({
        uglifyOptions: {
          output: {
            comments: /<\/?fs_premium_only>/i,
          },
        },
        extractComments: true,
      }),
    ],
  },
};
Stripping Paid Only PHP Functions
To add a function which will only be available in your paid version, simply add the __premium_only suffix to the function’s name. Make sure that all lines that call that method either directly, or by using hooks, are also wrapped in the premium only logic:
class My_Plugin {
    function init() {
        ...
         
        // This "if" block will be auto removed from the free version.
        if ( my_fs()->is__premium_only() ) {
            // Init premium version.
            $this->admin_init__premium_only();
         
            add_action( 'admin_init', array( &$this, 'admin_init_hook__premium_only' ) );
        }
         
        ...
    }
 
    // This method will be only included in the paid version.
    function admin_init__premium_only() {
        ...
    }
     
    // This method will be only included in the paid version.
    function admin_init_hook__premium_only() {
        ...
    }
}
Stripping Paid Only Files/Folders
There are two ways to exclude files from your free version:
- Add __premium_onlyjust before the file extension. For example:functions__premium_only.phpwill only be included in the premium plugin/theme version. This works for all type of files, not only for PHP.
- For plugins, add @fs_premium_only(a special meta tag) to the plugin's main PHP file header. Example:
<?php
/**
 * Plugin Name: My Awesome Plugin
 * Plugin URI:  http://my-awesome-plugin.com
 * Description: Create and manage Awesomeness right in WordPress.
 * Version:     1.0.0
 * Author:      Awesomattic
 * Author URI:  http://my-awesome-plugin.com/me/
 * License:     GPLv2
 * Text Domain: myplugin
 * Domain Path: /langs
 *
 * @fs_premium_only /lib/functions.php, /premium-files/
 */
 
if ( ! defined( 'ABSPATH' ) ) {
    exit;
}
 
// ... my code ...
The file /lib/functions.php and the directory /premium-files/ will be removed from the free plugin version.
- For themes, add @fs_premium_only, a special meta tag, to the theme's mainstyle.cssfile header. Example:
/**
 * Theme Name: My Awesome Theme
 * Theme URI:  http://my-awesome-theme.com
 * Description: Create and manage Awesomeness right in WordPress.
 * Version:     1.0.0
 * Author:      Awesomattic
 * Author URI:  http://my-awesome-theme.com/me/
 * License:     GPLv2
 * Text Domain: mytheme
 *
 * @fs_premium_only /premium-functions.php, /premium-files/
 */
 
body { ... }
span { ... }
Ignoring Folders Processing
If your product has a lot of code and/or many files, we created a special @fs_ignore meta tag to direct the PHP Preprocessor to ignore a given collection of files or folders that don't need processing. For example, if you have a vendors folder with 3rd party libraries that should remain untouched in both the free and paid versions, that folder would be a great candidate for the @fs_ignore tag.
For plugins, add @fs_ignore to the plugin's main PHP file header..
For themes, add @fs_ignore to the theme's style.css stylesheet.
This is similar to the @fs_premium_only example above).
Handling Premium readme.txt
In case you have a paid-only product (without a free version), the WordPress SDK will fetch it from our API instead of WordPress.org. For freemium plugins and themes, just like you only need to maintain a single version of your product, we've built a special markdown-friendly syntax to control which pieces of your readme.txt will show up on either the free or the premium version, or on both.
Paid-only readme.txt Content
If you wish to add content to appear only for the paid version, use the following syntax in your readme.txt:
[//]: # fs_premium_only_begin
Whatever goes in here will only appear in the paid readme
[//]: # fs_premium_only_end
Free-only readme.txt Content
For content to appear only inside the free version use the following syntax in your readme.txt:
[//]: # fs_free_only_begin
Whatever goes in here will be only in the free readme
[//]: # fs_free_only_end
Common readme.txt Content
For content to appear on both the free and the paid readme.txt files, you do not need to wrap it in any syntax.
Example of a Freemium readme.txt
If you deploy a freemium product with the following readme:
=== My Awesome Product ===
Requires at least: 3.0
Tested up to: 4.9
Stable tag: 3.1.0
License: GPLv2 or later
 
== Description ==
 
This is the best product ever!
 
== Installation ==
 
1. Upload the product to your blog.
2. Activate it.
[//]: # fs_premium_only_begin
3. Enter the license key you received after the purchase and activate it.
[//]: # fs_premium_only_end
 
== Change Log ==
 
[//]: # fs_premium_only_begin
= 1.0.0 =
* New: New premium feature!
[//]: # fs_premium_only_end
[//]: # fs_free_only_begin
= 1.0.0 =
* New: New free feature!
[//]: # fs_free_only_end
The free readme.txt will look like this:
=== My Awesome Product ===
Requires at least: 3.0
Tested up to: 4.9
Stable tag: 3.1.0
License: GPLv2 or later
 
== Description ==
 
This is the best product ever!
 
== Installation ==
 
1. Upload the product to your blog.
2. Activate it.
 
== Change Log ==
 
= 1.0.0 =
* New: New free feature!
The paid readme.txt will look like this:
Requires at least: 3.0
Tested up to: 4.9
Stable tag: 3.1.0
License: GPLv2 or later
 
== Description ==
 
This is the best product ever!
 
== Installation ==
 
1. Upload the product to your blog.
2. Activate it.
3. Enter the license key you received after the purchase and activate it.
 
== Change Log ==
 
= 1.0.0 =
* New: New paid feature!