Forums / Developer / Handling subscription based products

Handling subscription based products

Author Message

Eirik Alfstad Johansen

Wednesday 11 February 2004 6:16:18 am

Hi,

Since my company is both a web host and an Application Service Provider for web based software, I've come to discover the severe limitations of the ezprice datatype (and the rest of the shop module) for handling subscription based products and services. So, I though I would get a discussion going on this as I'm sure several people would benefit from such functionality.

Here are some ideas as to how subscription based products could be handled:

The following fields should be available when editing a class attribute created upon the ezprice datatype (in addition to the existing ones):

- The possibility to choose whether the product is a "Single purchase" or "Subscription based".
- For what period the product price is defined (1 day/week/month/year) (only applies if the price is "Subscription based").
- Whether the subscription is automatically renewed or not (only applies if the price is "Subscription based")

Next, it should be possible to create available subscription cycles under the "Shop" section of the admin interface. Each subscription cycle should consist of a number and a selection of the available options "days", "weeks", "months" and "years". Examples of subscription cycles would be "3 days" or "4 months" and these would be the available options when registering an order.

When displaying the price of a subscription based product (defined in the template for the view mode of the ezprice datatype), it should be visible for what period the price of the product is defined, for instance "$49 (pr/month)".

When registering an order, if the basket contains subscription based products, the shop/register template should automatically displays a drop down list containing the available subscription cycles. The ezorder table needs to be expanded in order to store the selected subscription cycle.

A new db table should be created to store information about each subscription cycle for an order. At a minimum, this new table would need to contain information about until which date the order has been "paid". (Of course, ezp doesn't know whether a product has been paid or not, so this will in fact be until which date the order has been ordered according to the subscription cycle and the order date, in addition to previous records in the table for the same order.)

A new cron job script would run to see if there are any subscription based orders that expire soon ("soon" would be defined as the amount of days in advance by an INI setting, probably in shop.ini).

For each subscription based order that expires soon

If the product is set to be automatically renewed
Store another subscription cycle for the order
Send info to admin containing information about the subscription renewal

If the product is not set to be automatically renewed
Send an email to the order owner (as fetched by the associated shopaccounthandler) asking them to renew their subscription, including a link to do so.

The page allowing a shopper to renew their subscription would require two new functions "all" and "own" to a new "renewsubscription" view of the shop module. If the user has the required privileges, a basket containing the products to be renewed (the subscription based products) will be displayed, including a button labeled "Renew". When the button is clicked, the same thing happens as when a product is set to be automatically renewed. In addition, an Email is sent to the order owner notifying them that the subscription has been renewed.

Finally, to have the possibility of using a credit card payment gateway upon the renewal of a subscription based order, the trigger "shop | renewsubscription | after" should be defined.

Questions/possible problems:

Another way to go would be to select a different set of available subscription cycles for each product object, and then select the preferred subscription cycle when adding a product to the cart. However, this would result in several subscription cycles for one order, and would make things more complicated, both for the shopper and the admin. Could anyone think of a scenario where product-specific subscription cycles are needed?

As far as I can tell, "ezproduct" would be a more appropriate name for the "ezprice" datatype, especially after adding the above functionality. Regardless, I prefer the "ezproduct" name as it much better describes the purpose of the datatype (and how it is handled in ezp).

Looking forward to hearing your input !

Sincerely,

Eirik Johansen

Sincerely,

Eirik Alfstad Johansen
http://www.netmaking.no/

Bruce Morrison

Tuesday 20 July 2004 7:07:23 pm

Hi Eirik

How are you currently handling this at the moment?

Cheers
Bruce

My Blog: http://www.stuffandcontent.com/
Follow me on twitter: http://twitter.com/brucemorrison
Consolidated eZ Publish Feed : http://friendfeed.com/rooms/ez-publish

Eirik Alfstad Johansen

Thursday 21 April 2005 1:04:31 am

Hi Bruce,

Just saw your question, about 9 months late. Don't know if you're still interested in the answer, but here goes.

I've created content objects for each product which contains information about when the product expires, what discount it has if any, for what length of time the product should be renewed, and whether or not it should be cancelled when it expires.

This information is all stored in separate attributes built upon default eZP datatypes. I then have a script set up as a cron job which loops through the objects and renews the ones (by incrementing the expiry date) that should be renewed and creates invoices for those.

This kind of works for us, but has of course a very low "recycling factor".

Sincerely,

Eirik Alfstad Johansen
http://www.netmaking.no/

Peter Scott

Thursday 19 June 2008 4:09:37 pm

Are there any changes to this in relation to Ez Publish 4.0 /Ez Flow 1.0?

Eirik Alfstad Johansen

Tuesday 24 June 2008 2:08:40 pm

Hi Peter,

As far as I know, there have been no improvements to the handling of subscription based products in eZ.

Since it's been 3 years since I last mentioned how we were handling this, I thought I would give anyone interested a little update. Here's the README file from the extension we ended up creating, and which still is in production today:

GENERAL OVERVIEW

The subscription extension creates a flexible framework for handling subscription.

It works by adding an attribute based upon the "Subscription" datatype to the content
class which stores your subscription data. The attribute will store the following data
associated with the subscription:

* Product count
* Product (a select list of all content objects on the site containing a price)
* Subscription cycle (in days, weeks, months or years)
* Subscription renewal date
* Subscription cancellation date (if any)
* Subscription discount (if any)

If you require additional information about a subscription, these can be stored as separate,
"normal" attributes.

Once you have created your subscriptions, the extension comes with several views for
presenting/handling the subscriptions. These views are accessible through the "Subscription"
tab which appears in your admin interface once the extension has been activated.

There is also an event type for creating a new subscription based on an order.

VIEWS

List subscriptions

Lists all subscriptions sorted by subscription renewal date. The view can be configured
display additional information about each subscription, fetched from either other attributes
of the subscription node, or from attributes of parent nodes.

Subscriptions which are due for renewal (a configurable x days until renewal) will appear in
red. When the "Renew" button is pushed, the renewal process is initiated. You can program your
own subscription handler which decides what happens when a subscription is renewed.

Turnover development

A simple, graphical representation of how the subscription revenue is growing, devided by
month.

Turnover detailed

All subscriptions grouped by subscription product and sorted by product generating the
highest recurring revenue.

Sincerely,

Eirik Alfstad Johansen
http://www.netmaking.no/

eZ debug

Timing: Jan 18 2025 11:45:25
Script start
Timing: Jan 18 2025 11:45:25
Module start 'content'
Timing: Jan 18 2025 11:45:26
Module end 'content'
Timing: Jan 18 2025 11:45:26
Script end

Main resources:

Total runtime0.9859 sec
Peak memory usage4,096.0000 KB
Database Queries202

Timing points:

CheckpointStart (sec)Duration (sec)Memory at start (KB)Memory used (KB)
Script start 0.00000.0086 587.7344180.8281
Module start 'content' 0.00860.8096 768.5625644.5938
Module end 'content' 0.81820.1676 1,413.1563345.3672
Script end 0.9858  1,758.5234 

Time accumulators:

 Accumulator Duration (sec) Duration (%) Count Average (sec)
Ini load
Load cache0.00440.4440210.0002
Check MTime0.00180.1852210.0001
Mysql Total
Database connection0.00110.108010.0011
Mysqli_queries0.892090.47782020.0044
Looping result0.00270.26952000.0000
Template Total0.948596.220.4742
Template load0.00280.281120.0014
Template processing0.945795.924220.4728
Template load and register function0.00010.009910.0001
states
state_id_array0.00120.121110.0012
state_identifier_array0.00130.128220.0006
Override
Cache load0.00250.2549810.0000
Sytem overhead
Fetch class attribute can translate value0.00150.153140.0004
Fetch class attribute name0.00140.142770.0002
XML
Image XML parsing0.00150.149740.0004
class_abstraction
Instantiating content class attribute0.00000.001890.0000
General
dbfile0.00290.2897330.0001
String conversion0.00000.000530.0000
Note: percentages do not add up to 100% because some accumulators overlap

CSS/JS files loaded with "ezjscPacker" during request:

CacheTypePacklevelSourceFiles
CSS0extension/community/design/community/stylesheets/ext/jquery.autocomplete.css
extension/community_design/design/suncana/stylesheets/scrollbars.css
extension/community_design/design/suncana/stylesheets/tabs.css
extension/community_design/design/suncana/stylesheets/roadmap.css
extension/community_design/design/suncana/stylesheets/content.css
extension/community_design/design/suncana/stylesheets/star-rating.css
extension/community_design/design/suncana/stylesheets/syntax_and_custom_tags.css
extension/community_design/design/suncana/stylesheets/buttons.css
extension/community_design/design/suncana/stylesheets/tweetbox.css
extension/community_design/design/suncana/stylesheets/jquery.fancybox-1.3.4.css
extension/bcsmoothgallery/design/standard/stylesheets/magnific-popup.css
extension/sevenx/design/simple/stylesheets/star_rating.css
extension/sevenx/design/simple/stylesheets/libs/fontawesome/css/all.min.css
extension/sevenx/design/simple/stylesheets/main.v02.css
extension/sevenx/design/simple/stylesheets/main.v02.res.css
JS0extension/ezjscore/design/standard/lib/yui/3.17.2/build/yui/yui-min.js
extension/ezjscore/design/standard/javascript/jquery-3.7.0.min.js
extension/community_design/design/suncana/javascript/jquery.ui.core.min.js
extension/community_design/design/suncana/javascript/jquery.ui.widget.min.js
extension/community_design/design/suncana/javascript/jquery.easing.1.3.js
extension/community_design/design/suncana/javascript/jquery.ui.tabs.js
extension/community_design/design/suncana/javascript/jquery.hoverIntent.min.js
extension/community_design/design/suncana/javascript/jquery.popmenu.js
extension/community_design/design/suncana/javascript/jScrollPane.js
extension/community_design/design/suncana/javascript/jquery.mousewheel.js
extension/community_design/design/suncana/javascript/jquery.cycle.all.js
extension/sevenx/design/simple/javascript/jquery.scrollTo.js
extension/community_design/design/suncana/javascript/jquery.cookie.js
extension/community_design/design/suncana/javascript/ezstarrating_jquery.js
extension/community_design/design/suncana/javascript/jquery.initboxes.js
extension/community_design/design/suncana/javascript/app.js
extension/community_design/design/suncana/javascript/twitterwidget.js
extension/community_design/design/suncana/javascript/community.js
extension/community_design/design/suncana/javascript/roadmap.js
extension/community_design/design/suncana/javascript/ez.js
extension/community_design/design/suncana/javascript/ezshareevents.js
extension/sevenx/design/simple/javascript/main.js

Templates used to render the page:

UsageRequested templateTemplateTemplate loadedEditOverride
1node/view/full.tplfull/forum_topic.tplextension/sevenx/design/simple/override/templates/full/forum_topic.tplEdit templateOverride template
4content/datatype/view/ezimage.tpl<No override>extension/sevenx/design/simple/templates/content/datatype/view/ezimage.tplEdit templateOverride template
5content/datatype/view/ezxmltext.tpl<No override>extension/community_design/design/suncana/templates/content/datatype/view/ezxmltext.tplEdit templateOverride template
17content/datatype/view/ezxmltags/paragraph.tpl<No override>extension/ezwebin/design/ezwebin/templates/content/datatype/view/ezxmltags/paragraph.tplEdit templateOverride template
12content/datatype/view/ezxmltags/line.tpl<No override>design/standard/templates/content/datatype/view/ezxmltags/line.tplEdit templateOverride template
1pagelayout.tpl<No override>extension/sevenx/design/simple/templates/pagelayout.tplEdit templateOverride template
 Number of times templates used: 40
 Number of unique templates used: 6

Time used to render debug report: 0.0001 secs