Merge pull request #2945 from coreos-inc/joseph.schorr/QS-53/new-pricing

New Quay.io pricing
This commit is contained in:
josephschorr 2017-12-18 13:21:49 -05:00 committed by GitHub
commit a829260091
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
5 changed files with 148 additions and 62 deletions

View file

@ -6,7 +6,7 @@ from calendar import timegm
from util.morecollections import AttrDict from util.morecollections import AttrDict
PLANS = [ PLANS = [
# Deprecated Plans # Deprecated Plans (2013-2014)
{ {
'title': 'Micro', 'title': 'Micro',
'price': 700, 'price': 700,
@ -16,7 +16,7 @@ PLANS = [
'bus_features': False, 'bus_features': False,
'deprecated': True, 'deprecated': True,
'free_trial_days': 14, 'free_trial_days': 14,
'superseded_by': None, 'superseded_by': 'personal-30',
'plans_page_hidden': False, 'plans_page_hidden': False,
}, },
{ {
@ -28,7 +28,7 @@ PLANS = [
'bus_features': False, 'bus_features': False,
'deprecated': True, 'deprecated': True,
'free_trial_days': 14, 'free_trial_days': 14,
'superseded_by': None, 'superseded_by': 'bus-micro-30',
'plans_page_hidden': False, 'plans_page_hidden': False,
}, },
{ {
@ -104,7 +104,105 @@ PLANS = [
'plans_page_hidden': False, 'plans_page_hidden': False,
}, },
# Active plans # Deprecated plans (2014-2017)
{
'title': 'Personal',
'price': 1200,
'privateRepos': 5,
'stripeId': 'personal-30',
'audience': 'Individuals',
'bus_features': False,
'deprecated': True,
'free_trial_days': 30,
'superseded_by': 'personal-2018',
'plans_page_hidden': False,
},
{
'title': 'Skiff',
'price': 2500,
'privateRepos': 10,
'stripeId': 'bus-micro-30',
'audience': 'For startups',
'bus_features': True,
'deprecated': True,
'free_trial_days': 30,
'superseded_by': 'bus-micro-2018',
'plans_page_hidden': False,
},
{
'title': 'Yacht',
'price': 5000,
'privateRepos': 20,
'stripeId': 'bus-small-30',
'audience': 'For small businesses',
'bus_features': True,
'deprecated': True,
'free_trial_days': 30,
'superseded_by': 'bus-small-2018',
'plans_page_hidden': False,
},
{
'title': 'Freighter',
'price': 10000,
'privateRepos': 50,
'stripeId': 'bus-medium-30',
'audience': 'For normal businesses',
'bus_features': True,
'deprecated': True,
'free_trial_days': 30,
'superseded_by': 'bus-medium-2018',
'plans_page_hidden': False,
},
{
'title': 'Tanker',
'price': 20000,
'privateRepos': 125,
'stripeId': 'bus-large-30',
'audience': 'For large businesses',
'bus_features': True,
'deprecated': True,
'free_trial_days': 30,
'superseded_by': 'bus-large-2018',
'plans_page_hidden': False,
},
{
'title': 'Carrier',
'price': 35000,
'privateRepos': 250,
'stripeId': 'bus-xlarge-30',
'audience': 'For extra large businesses',
'bus_features': True,
'deprecated': True,
'free_trial_days': 30,
'superseded_by': 'bus-xlarge-2018',
'plans_page_hidden': False,
},
{
'title': 'Huge',
'price': 65000,
'privateRepos': 500,
'stripeId': 'bus-500-30',
'audience': 'For huge business',
'bus_features': True,
'deprecated': True,
'free_trial_days': 30,
'superseded_by': 'bus-500-2018',
'plans_page_hidden': False,
},
{
'title': 'Huuge',
'price': 120000,
'privateRepos': 1000,
'stripeId': 'bus-1000-30',
'audience': 'For the SaaS savvy enterprise',
'bus_features': True,
'deprecated': True,
'free_trial_days': 30,
'superseded_by': 'bus-1000-2018',
'plans_page_hidden': False,
},
# Active plans (as of Dec 2017)
{ {
'title': 'Open Source', 'title': 'Open Source',
'price': 0, 'price': 0,
@ -118,10 +216,10 @@ PLANS = [
'plans_page_hidden': False, 'plans_page_hidden': False,
}, },
{ {
'title': 'Personal', 'title': 'Developer',
'price': 1200, 'price': 1500,
'privateRepos': 5, 'privateRepos': 5,
'stripeId': 'personal-30', 'stripeId': 'personal-2018',
'audience': 'Individuals', 'audience': 'Individuals',
'bus_features': False, 'bus_features': False,
'deprecated': False, 'deprecated': False,
@ -130,10 +228,10 @@ PLANS = [
'plans_page_hidden': False, 'plans_page_hidden': False,
}, },
{ {
'title': 'Skiff', 'title': 'Micro',
'price': 2500, 'price': 3000,
'privateRepos': 10, 'privateRepos': 10,
'stripeId': 'bus-micro-30', 'stripeId': 'bus-micro-2018',
'audience': 'For startups', 'audience': 'For startups',
'bus_features': True, 'bus_features': True,
'deprecated': False, 'deprecated': False,
@ -142,10 +240,10 @@ PLANS = [
'plans_page_hidden': False, 'plans_page_hidden': False,
}, },
{ {
'title': 'Yacht', 'title': 'Small',
'price': 5000, 'price': 6000,
'privateRepos': 20, 'privateRepos': 20,
'stripeId': 'bus-small-30', 'stripeId': 'bus-small-2018',
'audience': 'For small businesses', 'audience': 'For small businesses',
'bus_features': True, 'bus_features': True,
'deprecated': False, 'deprecated': False,
@ -154,10 +252,10 @@ PLANS = [
'plans_page_hidden': False, 'plans_page_hidden': False,
}, },
{ {
'title': 'Freighter', 'title': 'Medium',
'price': 10000, 'price': 12500,
'privateRepos': 50, 'privateRepos': 50,
'stripeId': 'bus-medium-30', 'stripeId': 'bus-medium-2018',
'audience': 'For normal businesses', 'audience': 'For normal businesses',
'bus_features': True, 'bus_features': True,
'deprecated': False, 'deprecated': False,
@ -166,10 +264,10 @@ PLANS = [
'plans_page_hidden': False, 'plans_page_hidden': False,
}, },
{ {
'title': 'Tanker', 'title': 'Large',
'price': 20000, 'price': 25000,
'privateRepos': 125, 'privateRepos': 125,
'stripeId': 'bus-large-30', 'stripeId': 'bus-large-2018',
'audience': 'For large businesses', 'audience': 'For large businesses',
'bus_features': True, 'bus_features': True,
'deprecated': False, 'deprecated': False,
@ -178,10 +276,10 @@ PLANS = [
'plans_page_hidden': False, 'plans_page_hidden': False,
}, },
{ {
'title': 'Carrier', 'title': 'Extra Large',
'price': 35000, 'price': 45000,
'privateRepos': 250, 'privateRepos': 250,
'stripeId': 'bus-xlarge-30', 'stripeId': 'bus-xlarge-2018',
'audience': 'For extra large businesses', 'audience': 'For extra large businesses',
'bus_features': True, 'bus_features': True,
'deprecated': False, 'deprecated': False,
@ -190,10 +288,10 @@ PLANS = [
'plans_page_hidden': False, 'plans_page_hidden': False,
}, },
{ {
'title': 'Huge', 'title': 'XXL',
'price': 65000, 'price': 85000,
'privateRepos': 500, 'privateRepos': 500,
'stripeId': 'bus-500-30', 'stripeId': 'bus-500-2018',
'audience': 'For huge business', 'audience': 'For huge business',
'bus_features': True, 'bus_features': True,
'deprecated': False, 'deprecated': False,
@ -202,10 +300,10 @@ PLANS = [
'plans_page_hidden': False, 'plans_page_hidden': False,
}, },
{ {
'title': 'Huuge', 'title': 'XXXL',
'price': 120000, 'price': 160000,
'privateRepos': 1000, 'privateRepos': 1000,
'stripeId': 'bus-1000-30', 'stripeId': 'bus-1000-2018',
'audience': 'For the SaaS savvy enterprise', 'audience': 'For the SaaS savvy enterprise',
'bus_features': True, 'bus_features': True,
'deprecated': False, 'deprecated': False,

View file

@ -62,7 +62,7 @@
</thead> </thead>
<tr ng-repeat="plan in plans" ng-show="isPlanVisible(plan, subscribedPlan)" <tr ng-repeat="plan in plans" ng-show="isPlanVisible(plan, subscribedPlan)"
ng-class="{'active':isPlanActive(plan, subscribedPlan), 'deprecated-plan':plan.deprecated}"> ng-class="{'active':isPlanActive(plan, subscribedPlan)}">
<td> <td>
{{ plan.title }} {{ plan.title }}
<div class="deprecated-plan-label" ng-show="plan.deprecated"> <div class="deprecated-plan-label" ng-show="plan.deprecated">

View file

@ -170,19 +170,19 @@
<!-- Plan columns --> <!-- Plan columns -->
<div class="row-container"> <div class="row-container">
<div class="row"> <div class="row">
<!-- Personal Plan --> <!-- Developer Plan -->
<div class="col-md-3 col-sm-6 plan-col"> <div class="col-md-3 col-sm-6 plan-col">
<div class="plan-box"> <div class="plan-box">
<div class="plan-header"> <div class="plan-header">
<span class="plan-box-price">$12/mo</span> <span class="plan-box-price">$15/mo</span>
<b>Personal</b> <b>Developer</b>
</div> </div>
<ul> <ul>
<li>5 private repositories</li> <li>5 private repositories</li>
<li>Unlimited public repos</li> <li>Unlimited public repos</li>
</ul> </ul>
<div class="plan-description">Great for individuals</div> <div class="plan-description">Great for individuals</div>
<a class="btn btn-primary trial-button" ng-click="buyNow('personal-30')"> <a class="btn btn-primary trial-button" ng-click="buyNow('personal-2018')">
Start Free Trial Start Free Trial
<i class="fa fa-angle-double-right"></i> <i class="fa fa-angle-double-right"></i>
</a> </a>
@ -193,12 +193,12 @@
</ul> </ul>
</div> </div>
<!-- Skiff Plan --> <!-- Micro Plan -->
<div class="col-md-3 col-sm-6 plan-col"> <div class="col-md-3 col-sm-6 plan-col">
<div class="plan-box"> <div class="plan-box">
<div class="plan-header"> <div class="plan-header">
<span class="plan-box-price">$25/mo</span> <span class="plan-box-price">$30/mo</span>
<b>Skiff</b> <b>Micro</b>
</div> </div>
<ul> <ul>
<li>10 private repositories</li> <li>10 private repositories</li>
@ -206,7 +206,7 @@
<li>Team-based permissions</li> <li>Team-based permissions</li>
</ul> </ul>
<div class="plan-description">Great for startups</div> <div class="plan-description">Great for startups</div>
<a class="btn btn-primary trial-button" ng-click="buyNow('bus-micro-30')"> <a class="btn btn-primary trial-button" ng-click="buyNow('bus-micro-2018')">
Start Free Trial Start Free Trial
<i class="fa fa-angle-double-right"></i> <i class="fa fa-angle-double-right"></i>
</a> </a>
@ -217,12 +217,12 @@
</ul> </ul>
</div> </div>
<!-- Yacht Plan --> <!-- Small Plan -->
<div class="col-md-3 col-sm-6 plan-col popular"> <div class="col-md-3 col-sm-6 plan-col popular">
<div class="plan-box"> <div class="plan-box">
<div class="plan-header"> <div class="plan-header">
<span class="plan-box-price">$50/mo</span> <span class="plan-box-price">$60/mo</span>
<b>Yacht</b> <b>Small</b>
</div> </div>
<ul> <ul>
<li>20 private repositories</li> <li>20 private repositories</li>
@ -230,7 +230,7 @@
<li>Team-based permissions</li> <li>Team-based permissions</li>
</ul> </ul>
<div class="plan-description">Great for small businesses</div> <div class="plan-description">Great for small businesses</div>
<a class="btn btn-primary trial-button" ng-click="buyNow('bus-small-30')"> <a class="btn btn-primary trial-button" ng-click="buyNow('bus-small-2018')">
Start Free Trial Start Free Trial
<i class="fa fa-angle-double-right"></i> <i class="fa fa-angle-double-right"></i>
</a> </a>

View file

@ -29,14 +29,8 @@ angular.module('quay').directive('planManager', function () {
return false; return false;
} }
// A plan is visible if it is not deprecated, or if it is both not superseded // A plan is visible if it is not deprecated, or if it is the namespace's current plan.
// by another plan, and also the active plan for the user.
if (plan['deprecated']) { if (plan['deprecated']) {
var superseded = PlanService.getPlanImmediately(plan['superseded_by']);
if (superseded) {
return !$scope.isPlanVisible(superseded, subscribedPlan);
}
return subscribedPlan && plan.stripeId === subscribedPlan.stripeId; return subscribedPlan && plan.stripeId === subscribedPlan.stripeId;
} }
@ -44,16 +38,10 @@ angular.module('quay').directive('planManager', function () {
}; };
$scope.isPlanActive = function(plan, subscribedPlan) { $scope.isPlanActive = function(plan, subscribedPlan) {
// A plan is active if it is either the plan directly subscribed to by the user
// or the plan which supersedes the plan to which the user is subscribed.
if (!subscribedPlan) { if (!subscribedPlan) {
return false; return false;
} }
if (subscribedPlan['deprecated'] && subscribedPlan['superseded_by']) {
return $scope.isPlanActive(plan, PlanService.getPlanImmediately(subscribedPlan['superseded_by']));
}
return plan.stripeId === subscribedPlan.stripeId; return plan.stripeId === subscribedPlan.stripeId;
}; };

View file

@ -440,14 +440,14 @@ class TestGetUserPrivateAllowed(ApiTestCase):
self.login(ADMIN_ACCESS_USER) self.login(ADMIN_ACCESS_USER)
# Change the subscription of the namespace. # Change the subscription of the namespace.
self.putJsonResponse(UserPlan, data=dict(plan='personal-30')) self.putJsonResponse(UserPlan, data=dict(plan='personal-2018'))
json = self.getJsonResponse(PrivateRepositories) json = self.getJsonResponse(PrivateRepositories)
assert json['privateCount'] >= 6 assert json['privateCount'] >= 6
assert not json['privateAllowed'] assert not json['privateAllowed']
# Change the subscription of the namespace. # Change the subscription of the namespace.
self.putJsonResponse(UserPlan, data=dict(plan='bus-large-30')) self.putJsonResponse(UserPlan, data=dict(plan='bus-large-2018'))
json = self.getJsonResponse(PrivateRepositories) json = self.getJsonResponse(PrivateRepositories)
assert json['privateAllowed'] assert json['privateAllowed']
@ -2019,7 +2019,7 @@ class TestChangeRepoVisibility(ApiTestCase):
self.assertEquals(True, json['is_public']) self.assertEquals(True, json['is_public'])
# Change the subscription of the namespace. # Change the subscription of the namespace.
self.putJsonResponse(UserPlan, data=dict(plan='personal-30')) self.putJsonResponse(UserPlan, data=dict(plan='personal-2018'))
# Try to make private. # Try to make private.
self.postJsonResponse(RepositoryVisibility, params=dict(repository=self.SIMPLE_REPO), self.postJsonResponse(RepositoryVisibility, params=dict(repository=self.SIMPLE_REPO),
@ -3166,11 +3166,11 @@ class TestUserSubscription(ApiTestCase):
self.assertEquals('free', sub['plan']) self.assertEquals('free', sub['plan'])
# Change the plan. # Change the plan.
self.putJsonResponse(UserPlan, data=dict(plan='bus-large-30')) self.putJsonResponse(UserPlan, data=dict(plan='bus-large-2018'))
# Verify # Verify
sub = self.getSubscription() sub = self.getSubscription()
self.assertEquals('bus-large-30', sub['plan']) self.assertEquals('bus-large-2018', sub['plan'])
class TestOrgSubscription(ApiTestCase): class TestOrgSubscription(ApiTestCase):
@ -3190,11 +3190,11 @@ class TestOrgSubscription(ApiTestCase):
# Change the plan. # Change the plan.
self.putJsonResponse(OrganizationPlan, params=dict(orgname=ORGANIZATION), self.putJsonResponse(OrganizationPlan, params=dict(orgname=ORGANIZATION),
data=dict(plan='bus-large-30')) data=dict(plan='bus-large-2018'))
# Verify # Verify
sub = self.getSubscription() sub = self.getSubscription()
self.assertEquals('bus-large-30', sub['plan']) self.assertEquals('bus-large-2018', sub['plan'])
class TestUserRobots(ApiTestCase): class TestUserRobots(ApiTestCase):