Turn off all references and API calls to billing if the feature is disabled
This commit is contained in:
parent
c374e8146a
commit
19a20a6c94
12 changed files with 135 additions and 62 deletions
|
@ -124,6 +124,9 @@ def format_date(date):
|
||||||
|
|
||||||
def add_method_metadata(name, value):
|
def add_method_metadata(name, value):
|
||||||
def modifier(func):
|
def modifier(func):
|
||||||
|
if func is None:
|
||||||
|
return None
|
||||||
|
|
||||||
if '__api_metadata' not in dir(func):
|
if '__api_metadata' not in dir(func):
|
||||||
func.__api_metadata = {}
|
func.__api_metadata = {}
|
||||||
func.__api_metadata[name] = value
|
func.__api_metadata[name] = value
|
||||||
|
@ -132,6 +135,9 @@ def add_method_metadata(name, value):
|
||||||
|
|
||||||
|
|
||||||
def method_metadata(func, name):
|
def method_metadata(func, name):
|
||||||
|
if func is None:
|
||||||
|
return None
|
||||||
|
|
||||||
if '__api_metadata' in dir(func):
|
if '__api_metadata' in dir(func):
|
||||||
return func.__api_metadata.get(name, None)
|
return func.__api_metadata.get(name, None)
|
||||||
return None
|
return None
|
||||||
|
|
|
@ -4,13 +4,14 @@ from flask import request
|
||||||
|
|
||||||
from endpoints.api import (resource, nickname, ApiResource, validate_json_request, log_action,
|
from endpoints.api import (resource, nickname, ApiResource, validate_json_request, log_action,
|
||||||
related_user_resource, internal_only, Unauthorized, NotFound,
|
related_user_resource, internal_only, Unauthorized, NotFound,
|
||||||
require_user_admin)
|
require_user_admin, show_if, hide_if)
|
||||||
from endpoints.api.subscribe import subscribe, subscription_view
|
from endpoints.api.subscribe import subscribe, subscription_view
|
||||||
from auth.permissions import AdministerOrganizationPermission
|
from auth.permissions import AdministerOrganizationPermission
|
||||||
from auth.auth_context import get_authenticated_user
|
from auth.auth_context import get_authenticated_user
|
||||||
from data import model
|
from data import model
|
||||||
from data.plans import PLANS
|
from data.plans import PLANS
|
||||||
|
|
||||||
|
import features
|
||||||
|
|
||||||
def carderror_response(e):
|
def carderror_response(e):
|
||||||
return {'carderror': e.message}, 402
|
return {'carderror': e.message}, 402
|
||||||
|
@ -79,6 +80,7 @@ def get_invoices(customer_id):
|
||||||
|
|
||||||
|
|
||||||
@resource('/v1/plans/')
|
@resource('/v1/plans/')
|
||||||
|
@show_if(features.BILLING)
|
||||||
class ListPlans(ApiResource):
|
class ListPlans(ApiResource):
|
||||||
""" Resource for listing the available plans. """
|
""" Resource for listing the available plans. """
|
||||||
@nickname('listPlans')
|
@nickname('listPlans')
|
||||||
|
@ -91,6 +93,7 @@ class ListPlans(ApiResource):
|
||||||
|
|
||||||
@resource('/v1/user/card')
|
@resource('/v1/user/card')
|
||||||
@internal_only
|
@internal_only
|
||||||
|
@show_if(features.BILLING)
|
||||||
class UserCard(ApiResource):
|
class UserCard(ApiResource):
|
||||||
""" Resource for managing a user's credit card. """
|
""" Resource for managing a user's credit card. """
|
||||||
schemas = {
|
schemas = {
|
||||||
|
@ -132,6 +135,7 @@ class UserCard(ApiResource):
|
||||||
@resource('/v1/organization/<orgname>/card')
|
@resource('/v1/organization/<orgname>/card')
|
||||||
@internal_only
|
@internal_only
|
||||||
@related_user_resource(UserCard)
|
@related_user_resource(UserCard)
|
||||||
|
@show_if(features.BILLING)
|
||||||
class OrganizationCard(ApiResource):
|
class OrganizationCard(ApiResource):
|
||||||
""" Resource for managing an organization's credit card. """
|
""" Resource for managing an organization's credit card. """
|
||||||
schemas = {
|
schemas = {
|
||||||
|
@ -178,6 +182,7 @@ class OrganizationCard(ApiResource):
|
||||||
|
|
||||||
@resource('/v1/user/plan')
|
@resource('/v1/user/plan')
|
||||||
@internal_only
|
@internal_only
|
||||||
|
@show_if(features.BILLING)
|
||||||
class UserPlan(ApiResource):
|
class UserPlan(ApiResource):
|
||||||
""" Resource for managing a user's subscription. """
|
""" Resource for managing a user's subscription. """
|
||||||
schemas = {
|
schemas = {
|
||||||
|
@ -234,6 +239,7 @@ class UserPlan(ApiResource):
|
||||||
@resource('/v1/organization/<orgname>/plan')
|
@resource('/v1/organization/<orgname>/plan')
|
||||||
@internal_only
|
@internal_only
|
||||||
@related_user_resource(UserPlan)
|
@related_user_resource(UserPlan)
|
||||||
|
@show_if(features.BILLING)
|
||||||
class OrganizationPlan(ApiResource):
|
class OrganizationPlan(ApiResource):
|
||||||
""" Resource for managing a org's subscription. """
|
""" Resource for managing a org's subscription. """
|
||||||
schemas = {
|
schemas = {
|
||||||
|
@ -294,6 +300,7 @@ class OrganizationPlan(ApiResource):
|
||||||
|
|
||||||
@resource('/v1/user/invoices')
|
@resource('/v1/user/invoices')
|
||||||
@internal_only
|
@internal_only
|
||||||
|
@show_if(features.BILLING)
|
||||||
class UserInvoiceList(ApiResource):
|
class UserInvoiceList(ApiResource):
|
||||||
""" Resource for listing a user's invoices. """
|
""" Resource for listing a user's invoices. """
|
||||||
@require_user_admin
|
@require_user_admin
|
||||||
|
@ -310,6 +317,7 @@ class UserInvoiceList(ApiResource):
|
||||||
@resource('/v1/organization/<orgname>/invoices')
|
@resource('/v1/organization/<orgname>/invoices')
|
||||||
@internal_only
|
@internal_only
|
||||||
@related_user_resource(UserInvoiceList)
|
@related_user_resource(UserInvoiceList)
|
||||||
|
@show_if(features.BILLING)
|
||||||
class OrgnaizationInvoiceList(ApiResource):
|
class OrgnaizationInvoiceList(ApiResource):
|
||||||
""" Resource for listing an orgnaization's invoices. """
|
""" Resource for listing an orgnaization's invoices. """
|
||||||
@nickname('listOrgInvoices')
|
@nickname('listOrgInvoices')
|
||||||
|
|
|
@ -5,7 +5,7 @@ from flask import request
|
||||||
|
|
||||||
from endpoints.api import (resource, nickname, ApiResource, validate_json_request, request_error,
|
from endpoints.api import (resource, nickname, ApiResource, validate_json_request, request_error,
|
||||||
related_user_resource, internal_only, Unauthorized, NotFound,
|
related_user_resource, internal_only, Unauthorized, NotFound,
|
||||||
require_user_admin, log_action)
|
require_user_admin, log_action, show_if)
|
||||||
from endpoints.api.team import team_view
|
from endpoints.api.team import team_view
|
||||||
from endpoints.api.user import User, PrivateRepositories
|
from endpoints.api.user import User, PrivateRepositories
|
||||||
from auth.permissions import (AdministerOrganizationPermission, OrganizationMemberPermission,
|
from auth.permissions import (AdministerOrganizationPermission, OrganizationMemberPermission,
|
||||||
|
@ -15,6 +15,8 @@ from data import model
|
||||||
from data.plans import get_plan
|
from data.plans import get_plan
|
||||||
from util.gravatar import compute_hash
|
from util.gravatar import compute_hash
|
||||||
|
|
||||||
|
import features
|
||||||
|
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
@ -163,6 +165,7 @@ class Organization(ApiResource):
|
||||||
@resource('/v1/organization/<orgname>/private')
|
@resource('/v1/organization/<orgname>/private')
|
||||||
@internal_only
|
@internal_only
|
||||||
@related_user_resource(PrivateRepositories)
|
@related_user_resource(PrivateRepositories)
|
||||||
|
@show_if(features.BILLING)
|
||||||
class OrgPrivateRepositories(ApiResource):
|
class OrgPrivateRepositories(ApiResource):
|
||||||
""" Custom verb to compute whether additional private repositories are available. """
|
""" Custom verb to compute whether additional private repositories are available. """
|
||||||
@nickname('getOrganizationPrivateAllowed')
|
@nickname('getOrganizationPrivateAllowed')
|
||||||
|
|
|
@ -6,6 +6,7 @@ from endpoints.common import check_repository_usage
|
||||||
from data import model
|
from data import model
|
||||||
from data.plans import PLANS
|
from data.plans import PLANS
|
||||||
|
|
||||||
|
import features
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
@ -24,6 +25,9 @@ def subscription_view(stripe_subscription, used_repos):
|
||||||
|
|
||||||
|
|
||||||
def subscribe(user, plan, token, require_business_plan):
|
def subscribe(user, plan, token, require_business_plan):
|
||||||
|
if not features.BILLING:
|
||||||
|
return
|
||||||
|
|
||||||
plan_found = None
|
plan_found = None
|
||||||
for plan_obj in PLANS:
|
for plan_obj in PLANS:
|
||||||
if plan_obj['stripeId'] == plan:
|
if plan_obj['stripeId'] == plan:
|
||||||
|
|
|
@ -194,6 +194,7 @@ class User(ApiResource):
|
||||||
|
|
||||||
@resource('/v1/user/private')
|
@resource('/v1/user/private')
|
||||||
@internal_only
|
@internal_only
|
||||||
|
@show_if(features.BILLING)
|
||||||
class PrivateRepositories(ApiResource):
|
class PrivateRepositories(ApiResource):
|
||||||
""" Operations dealing with the available count of private repositories. """
|
""" Operations dealing with the available count of private repositories. """
|
||||||
@require_user_admin
|
@require_user_admin
|
||||||
|
@ -249,8 +250,7 @@ class ConvertToOrganization(ApiResource):
|
||||||
'description': 'Information required to convert a user to an organization.',
|
'description': 'Information required to convert a user to an organization.',
|
||||||
'required': [
|
'required': [
|
||||||
'adminUser',
|
'adminUser',
|
||||||
'adminPassword',
|
'adminPassword'
|
||||||
'plan',
|
|
||||||
],
|
],
|
||||||
'properties': {
|
'properties': {
|
||||||
'adminUser': {
|
'adminUser': {
|
||||||
|
@ -263,7 +263,7 @@ class ConvertToOrganization(ApiResource):
|
||||||
},
|
},
|
||||||
'plan': {
|
'plan': {
|
||||||
'type': 'string',
|
'type': 'string',
|
||||||
'description': 'The plan to which the organizatino should be subscribed',
|
'description': 'The plan to which the organization should be subscribed',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
@ -290,7 +290,8 @@ class ConvertToOrganization(ApiResource):
|
||||||
message='The admin user credentials are not valid')
|
message='The admin user credentials are not valid')
|
||||||
|
|
||||||
# Subscribe the organization to the new plan.
|
# Subscribe the organization to the new plan.
|
||||||
plan = convert_data['plan']
|
if features.BILLING:
|
||||||
|
plan = convert_data.get('plan', 'free')
|
||||||
subscribe(user, plan, None, True) # Require business plans
|
subscribe(user, plan, None, True) # Require business plans
|
||||||
|
|
||||||
# Convert the user to an organization.
|
# Convert the user to an organization.
|
||||||
|
|
|
@ -20,6 +20,8 @@ from util.names import parse_repository_name
|
||||||
from util.gravatar import compute_hash
|
from util.gravatar import compute_hash
|
||||||
from auth import scopes
|
from auth import scopes
|
||||||
|
|
||||||
|
import features
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
web = Blueprint('web', __name__)
|
web = Blueprint('web', __name__)
|
||||||
|
@ -54,6 +56,7 @@ def snapshot(path = ''):
|
||||||
|
|
||||||
@web.route('/plans/')
|
@web.route('/plans/')
|
||||||
@no_cache
|
@no_cache
|
||||||
|
@route_show_if(features.BILLING)
|
||||||
def plans():
|
def plans():
|
||||||
return index('')
|
return index('')
|
||||||
|
|
||||||
|
@ -152,6 +155,7 @@ def privacy():
|
||||||
|
|
||||||
|
|
||||||
@web.route('/receipt', methods=['GET'])
|
@web.route('/receipt', methods=['GET'])
|
||||||
|
@route_show_if(features.BILLING)
|
||||||
def receipt():
|
def receipt():
|
||||||
if not current_user.is_authenticated():
|
if not current_user.is_authenticated():
|
||||||
abort(401)
|
abort(401)
|
||||||
|
|
|
@ -14,7 +14,7 @@
|
||||||
<li><a ng-href="/repository/" target="{{ appLinkTarget() }}">Repositories</a></li>
|
<li><a ng-href="/repository/" target="{{ appLinkTarget() }}">Repositories</a></li>
|
||||||
<li><a href="http://docs.quay.io/" target="_blank">Docs</a></li>
|
<li><a href="http://docs.quay.io/" target="_blank">Docs</a></li>
|
||||||
<li><a ng-href="/tutorial/" target="{{ appLinkTarget() }}">Tutorial</a></li>
|
<li><a ng-href="/tutorial/" target="{{ appLinkTarget() }}">Tutorial</a></li>
|
||||||
<li><a ng-href="/plans/" target="{{ appLinkTarget() }}">Pricing</a></li>
|
<li><a ng-href="/plans/" target="{{ appLinkTarget() }}" quay-require="['BILLING']">Pricing</a></li>
|
||||||
<li><a ng-href="/organizations/" target="{{ appLinkTarget() }}">Organizations</a></li>
|
<li><a ng-href="/organizations/" target="{{ appLinkTarget() }}">Organizations</a></li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
|
|
|
@ -876,8 +876,8 @@ quayApp = angular.module('quay', ['ngRoute', 'chieffancypants.loadingBar', 'angu
|
||||||
return keyService;
|
return keyService;
|
||||||
}]);
|
}]);
|
||||||
|
|
||||||
$provide.factory('PlanService', ['KeyService', 'UserService', 'CookieService', 'ApiService',
|
$provide.factory('PlanService', ['KeyService', 'UserService', 'CookieService', 'ApiService', 'Features',
|
||||||
function(KeyService, UserService, CookieService, ApiService) {
|
function(KeyService, UserService, CookieService, ApiService, Features) {
|
||||||
var plans = null;
|
var plans = null;
|
||||||
var planDict = {};
|
var planDict = {};
|
||||||
var planService = {};
|
var planService = {};
|
||||||
|
@ -903,7 +903,9 @@ quayApp = angular.module('quay', ['ngRoute', 'chieffancypants.loadingBar', 'angu
|
||||||
};
|
};
|
||||||
|
|
||||||
planService.notePlan = function(planId) {
|
planService.notePlan = function(planId) {
|
||||||
|
if (Features.BILLING) {
|
||||||
CookieService.putSession('quay.notedplan', planId);
|
CookieService.putSession('quay.notedplan', planId);
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
planService.isOrgCompatible = function(plan) {
|
planService.isOrgCompatible = function(plan) {
|
||||||
|
@ -929,7 +931,7 @@ quayApp = angular.module('quay', ['ngRoute', 'chieffancypants.loadingBar', 'angu
|
||||||
|
|
||||||
planService.handleNotedPlan = function() {
|
planService.handleNotedPlan = function() {
|
||||||
var planId = planService.getAndResetNotedPlan();
|
var planId = planService.getAndResetNotedPlan();
|
||||||
if (!planId) { return false; }
|
if (!planId || !Features.BILLING) { return false; }
|
||||||
|
|
||||||
UserService.load(function() {
|
UserService.load(function() {
|
||||||
if (UserService.currentUser().anonymous) {
|
if (UserService.currentUser().anonymous) {
|
||||||
|
@ -974,6 +976,8 @@ quayApp = angular.module('quay', ['ngRoute', 'chieffancypants.loadingBar', 'angu
|
||||||
};
|
};
|
||||||
|
|
||||||
planService.verifyLoaded = function(callback) {
|
planService.verifyLoaded = function(callback) {
|
||||||
|
if (!Features.BILLING) { return; }
|
||||||
|
|
||||||
if (plans) {
|
if (plans) {
|
||||||
callback(plans);
|
callback(plans);
|
||||||
return;
|
return;
|
||||||
|
@ -1033,10 +1037,14 @@ quayApp = angular.module('quay', ['ngRoute', 'chieffancypants.loadingBar', 'angu
|
||||||
};
|
};
|
||||||
|
|
||||||
planService.getSubscription = function(orgname, success, failure) {
|
planService.getSubscription = function(orgname, success, failure) {
|
||||||
|
if (!Features.BILLING) { return; }
|
||||||
|
|
||||||
ApiService.getSubscription(orgname).then(success, failure);
|
ApiService.getSubscription(orgname).then(success, failure);
|
||||||
};
|
};
|
||||||
|
|
||||||
planService.setSubscription = function(orgname, planId, success, failure, opt_token) {
|
planService.setSubscription = function(orgname, planId, success, failure, opt_token) {
|
||||||
|
if (!Features.BILLING) { return; }
|
||||||
|
|
||||||
var subscriptionDetails = {
|
var subscriptionDetails = {
|
||||||
plan: planId
|
plan: planId
|
||||||
};
|
};
|
||||||
|
@ -1056,6 +1064,8 @@ quayApp = angular.module('quay', ['ngRoute', 'chieffancypants.loadingBar', 'angu
|
||||||
};
|
};
|
||||||
|
|
||||||
planService.getCardInfo = function(orgname, callback) {
|
planService.getCardInfo = function(orgname, callback) {
|
||||||
|
if (!Features.BILLING) { return; }
|
||||||
|
|
||||||
ApiService.getCard(orgname).then(function(resp) {
|
ApiService.getCard(orgname).then(function(resp) {
|
||||||
callback(resp.card);
|
callback(resp.card);
|
||||||
}, function() {
|
}, function() {
|
||||||
|
@ -1064,6 +1074,8 @@ quayApp = angular.module('quay', ['ngRoute', 'chieffancypants.loadingBar', 'angu
|
||||||
};
|
};
|
||||||
|
|
||||||
planService.changePlan = function($scope, orgname, planId, callbacks) {
|
planService.changePlan = function($scope, orgname, planId, callbacks) {
|
||||||
|
if (!Features.BILLING) { return; }
|
||||||
|
|
||||||
if (callbacks['started']) {
|
if (callbacks['started']) {
|
||||||
callbacks['started']();
|
callbacks['started']();
|
||||||
}
|
}
|
||||||
|
@ -1089,6 +1101,8 @@ quayApp = angular.module('quay', ['ngRoute', 'chieffancypants.loadingBar', 'angu
|
||||||
};
|
};
|
||||||
|
|
||||||
planService.changeCreditCard = function($scope, orgname, callbacks) {
|
planService.changeCreditCard = function($scope, orgname, callbacks) {
|
||||||
|
if (!Features.BILLING) { return; }
|
||||||
|
|
||||||
if (callbacks['opening']) {
|
if (callbacks['opening']) {
|
||||||
callbacks['opening']();
|
callbacks['opening']();
|
||||||
}
|
}
|
||||||
|
@ -1145,6 +1159,8 @@ quayApp = angular.module('quay', ['ngRoute', 'chieffancypants.loadingBar', 'angu
|
||||||
};
|
};
|
||||||
|
|
||||||
planService.showSubscribeDialog = function($scope, orgname, planId, callbacks) {
|
planService.showSubscribeDialog = function($scope, orgname, planId, callbacks) {
|
||||||
|
if (!Features.BILLING) { return; }
|
||||||
|
|
||||||
if (callbacks['opening']) {
|
if (callbacks['opening']) {
|
||||||
callbacks['opening']();
|
callbacks['opening']();
|
||||||
}
|
}
|
||||||
|
|
|
@ -1607,7 +1607,9 @@ function RepoAdminCtrl($scope, Restangular, ApiService, KeyService, $routeParams
|
||||||
}
|
}
|
||||||
|
|
||||||
function UserAdminCtrl($scope, $timeout, $location, ApiService, PlanService, UserService, CookieService, KeyService,
|
function UserAdminCtrl($scope, $timeout, $location, ApiService, PlanService, UserService, CookieService, KeyService,
|
||||||
$routeParams, $http) {
|
$routeParams, $http, Features) {
|
||||||
|
$scope.Features = Features;
|
||||||
|
|
||||||
if ($routeParams['migrate']) {
|
if ($routeParams['migrate']) {
|
||||||
$('#migrateTab').tab('show')
|
$('#migrateTab').tab('show')
|
||||||
}
|
}
|
||||||
|
@ -1690,6 +1692,7 @@ function UserAdminCtrl($scope, $timeout, $location, ApiService, PlanService, Use
|
||||||
};
|
};
|
||||||
|
|
||||||
$scope.showConvertForm = function() {
|
$scope.showConvertForm = function() {
|
||||||
|
if (Features.BILLING) {
|
||||||
PlanService.getMatchingBusinessPlan(function(plan) {
|
PlanService.getMatchingBusinessPlan(function(plan) {
|
||||||
$scope.org.plan = plan;
|
$scope.org.plan = plan;
|
||||||
});
|
});
|
||||||
|
@ -1697,6 +1700,7 @@ function UserAdminCtrl($scope, $timeout, $location, ApiService, PlanService, Use
|
||||||
PlanService.getPlans(function(plans) {
|
PlanService.getPlans(function(plans) {
|
||||||
$scope.orgPlans = plans;
|
$scope.orgPlans = plans;
|
||||||
});
|
});
|
||||||
|
}
|
||||||
|
|
||||||
$scope.convertStep = 1;
|
$scope.convertStep = 1;
|
||||||
};
|
};
|
||||||
|
@ -1711,7 +1715,7 @@ function UserAdminCtrl($scope, $timeout, $location, ApiService, PlanService, Use
|
||||||
var data = {
|
var data = {
|
||||||
'adminUser': $scope.org.adminUser,
|
'adminUser': $scope.org.adminUser,
|
||||||
'adminPassword': $scope.org.adminPassword,
|
'adminPassword': $scope.org.adminPassword,
|
||||||
'plan': $scope.org.plan.stripeId
|
'plan': $scope.org.plan ? $scope.org.plan.stripeId : ''
|
||||||
};
|
};
|
||||||
|
|
||||||
ApiService.convertUserToOrganization(data).then(function(resp) {
|
ApiService.convertUserToOrganization(data).then(function(resp) {
|
||||||
|
@ -1912,7 +1916,7 @@ function V1Ctrl($scope, $location, UserService) {
|
||||||
UserService.updateUserIn($scope);
|
UserService.updateUserIn($scope);
|
||||||
}
|
}
|
||||||
|
|
||||||
function NewRepoCtrl($scope, $location, $http, $timeout, UserService, ApiService, PlanService, KeyService) {
|
function NewRepoCtrl($scope, $location, $http, $timeout, UserService, ApiService, PlanService, KeyService, Features) {
|
||||||
UserService.updateUserIn($scope);
|
UserService.updateUserIn($scope);
|
||||||
|
|
||||||
$scope.githubRedirectUri = KeyService.githubRedirectUri;
|
$scope.githubRedirectUri = KeyService.githubRedirectUri;
|
||||||
|
@ -2034,6 +2038,12 @@ function NewRepoCtrl($scope, $location, $http, $timeout, UserService, ApiService
|
||||||
var checkPrivateAllowed = function() {
|
var checkPrivateAllowed = function() {
|
||||||
if (!$scope.repo || !$scope.repo.namespace) { return; }
|
if (!$scope.repo || !$scope.repo.namespace) { return; }
|
||||||
|
|
||||||
|
if (!Features.BILLING) {
|
||||||
|
$scope.checkingPlan = false;
|
||||||
|
$scope.planRequired = null;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
$scope.checkingPlan = true;
|
$scope.checkingPlan = true;
|
||||||
|
|
||||||
var isUserNamespace = $scope.isUserNamespace;
|
var isUserNamespace = $scope.isUserNamespace;
|
||||||
|
@ -2160,10 +2170,11 @@ function OrgViewCtrl($rootScope, $scope, ApiService, $routeParams) {
|
||||||
loadOrganization();
|
loadOrganization();
|
||||||
}
|
}
|
||||||
|
|
||||||
function OrgAdminCtrl($rootScope, $scope, $timeout, Restangular, $routeParams, UserService, PlanService, ApiService) {
|
function OrgAdminCtrl($rootScope, $scope, $timeout, Restangular, $routeParams, UserService, PlanService, ApiService, Features) {
|
||||||
var orgname = $routeParams.orgname;
|
var orgname = $routeParams.orgname;
|
||||||
|
|
||||||
// Load the list of plans.
|
// Load the list of plans.
|
||||||
|
if (Features.BILLING) {
|
||||||
PlanService.getPlans(function(plans) {
|
PlanService.getPlans(function(plans) {
|
||||||
$scope.plans = plans;
|
$scope.plans = plans;
|
||||||
$scope.plan_map = {};
|
$scope.plan_map = {};
|
||||||
|
@ -2172,6 +2183,7 @@ function OrgAdminCtrl($rootScope, $scope, $timeout, Restangular, $routeParams, U
|
||||||
$scope.plan_map[plans[i].stripeId] = plans[i];
|
$scope.plan_map[plans[i].stripeId] = plans[i];
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
}
|
||||||
|
|
||||||
$scope.orgname = orgname;
|
$scope.orgname = orgname;
|
||||||
$scope.membersLoading = true;
|
$scope.membersLoading = true;
|
||||||
|
@ -2354,11 +2366,15 @@ function OrgsCtrl($scope, UserService) {
|
||||||
browserchrome.update();
|
browserchrome.update();
|
||||||
}
|
}
|
||||||
|
|
||||||
function NewOrgCtrl($scope, $routeParams, $timeout, $location, UserService, PlanService, ApiService, CookieService) {
|
function NewOrgCtrl($scope, $routeParams, $timeout, $location, UserService, PlanService, ApiService, CookieService, Features) {
|
||||||
|
$scope.Features = Features;
|
||||||
|
$scope.holder = {};
|
||||||
|
|
||||||
UserService.updateUserIn($scope);
|
UserService.updateUserIn($scope);
|
||||||
|
|
||||||
var requested = $routeParams['plan'];
|
var requested = $routeParams['plan'];
|
||||||
|
|
||||||
|
if (Features.BILLING) {
|
||||||
// Load the list of plans.
|
// Load the list of plans.
|
||||||
PlanService.getPlans(function(plans) {
|
PlanService.getPlans(function(plans) {
|
||||||
$scope.plans = plans;
|
$scope.plans = plans;
|
||||||
|
@ -2369,15 +2385,20 @@ function NewOrgCtrl($scope, $routeParams, $timeout, $location, UserService, Plan
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
}
|
||||||
|
|
||||||
$scope.signedIn = function() {
|
$scope.signedIn = function() {
|
||||||
|
if (Features.BILLING) {
|
||||||
PlanService.handleNotedPlan();
|
PlanService.handleNotedPlan();
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
$scope.signinStarted = function() {
|
$scope.signinStarted = function() {
|
||||||
|
if (Features.BILLING) {
|
||||||
PlanService.getMinimumPlan(1, true, function(plan) {
|
PlanService.getMinimumPlan(1, true, function(plan) {
|
||||||
PlanService.notePlan(plan.stripeId);
|
PlanService.notePlan(plan.stripeId);
|
||||||
});
|
});
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
$scope.setPlan = function(plan) {
|
$scope.setPlan = function(plan) {
|
||||||
|
@ -2409,7 +2430,7 @@ function NewOrgCtrl($scope, $routeParams, $timeout, $location, UserService, Plan
|
||||||
};
|
};
|
||||||
|
|
||||||
// If the selected plan is free, simply move to the org page.
|
// If the selected plan is free, simply move to the org page.
|
||||||
if ($scope.currentPlan.price == 0) {
|
if (!Features.BILLING || $scope.currentPlan.price == 0) {
|
||||||
showOrg();
|
showOrg();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
|
@ -66,13 +66,14 @@
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<!-- Plans Table -->
|
<!-- Plans Table -->
|
||||||
<div class="form-group nested plan-group">
|
<div class="form-group nested plan-group" quay-require="['BILLING']">
|
||||||
<strong>Choose your organization's plan</strong>
|
<strong>Choose your organization's plan</strong>
|
||||||
<div class="plans-table" plans="plans" current-plan="currentPlan"></div>
|
<div class="plans-table" plans="plans" current-plan="holder.currentPlan"></div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="button-bar">
|
<div class="button-bar">
|
||||||
<button class="btn btn-large btn-success" type="submit" ng-disabled="newOrgForm.$invalid || !currentPlan"
|
<button class="btn btn-large btn-success" type="submit"
|
||||||
|
ng-disabled="newOrgForm.$invalid || (Features.BILLING && !holder.currentPlan)"
|
||||||
analytics-on analytics-event="create_organization">
|
analytics-on analytics-event="create_organization">
|
||||||
Create Organization
|
Create Organization
|
||||||
</button>
|
</button>
|
||||||
|
|
|
@ -6,15 +6,23 @@
|
||||||
<!-- Side tabs -->
|
<!-- Side tabs -->
|
||||||
<div class="col-md-2">
|
<div class="col-md-2">
|
||||||
<ul class="nav nav-pills nav-stacked">
|
<ul class="nav nav-pills nav-stacked">
|
||||||
<li class="active"><a href="javascript:void(0)" data-toggle="tab" data-target="#plan">Plan and Usage</a></li>
|
<li class="active" quay-require="['BILLING']">
|
||||||
<li><a href="javascript:void(0)" data-toggle="tab" data-target="#settings">Organization Settings</a></li>
|
<a href="javascript:void(0)" data-toggle="tab" data-target="#plan">Plan and Usage</a>
|
||||||
|
</li>
|
||||||
|
<li quay-classes="{'!BILLING': 'active'}">
|
||||||
|
<a href="javascript:void(0)" data-toggle="tab" data-target="#settings">Organization Settings</a>
|
||||||
|
</li>
|
||||||
<li><a href="javascript:void(0)" data-toggle="tab" data-target="#logs" ng-click="loadLogs()">Usage Logs</a></li>
|
<li><a href="javascript:void(0)" data-toggle="tab" data-target="#logs" ng-click="loadLogs()">Usage Logs</a></li>
|
||||||
<li><a href="javascript:void(0)" data-toggle="tab" data-target="#members" ng-click="loadMembers()">Members</a></li>
|
<li><a href="javascript:void(0)" data-toggle="tab" data-target="#members" ng-click="loadMembers()">Members</a></li>
|
||||||
<li><a href="javascript:void(0)" data-toggle="tab" data-target="#robots">Robot Accounts</a></li>
|
<li><a href="javascript:void(0)" data-toggle="tab" data-target="#robots">Robot Accounts</a></li>
|
||||||
<li><a href="javascript:void(0)" data-toggle="tab" data-target="#prototypes">Default Permissions</a></li>
|
<li><a href="javascript:void(0)" data-toggle="tab" data-target="#prototypes">Default Permissions</a></li>
|
||||||
<li><a href="javascript:void(0)" data-toggle="tab" data-target="#applications" ng-click="loadApplications()">Applications</a></li>
|
<li><a href="javascript:void(0)" data-toggle="tab" data-target="#applications" ng-click="loadApplications()">Applications</a></li>
|
||||||
<li ng-show="hasPaidPlan"><a href="javascript:void(0)" data-toggle="tab" data-target="#billingoptions">Billing</a></li>
|
<li ng-show="hasPaidPlan" quay-require="['BILLING']">
|
||||||
<li ng-show="hasPaidPlan"><a href="javascript:void(0)" data-toggle="tab" data-target="#billing" ng-click="loadInvoices()">Billing History</a></li>
|
<a href="javascript:void(0)" data-toggle="tab" data-target="#billingoptions">Billing</a>
|
||||||
|
</li>
|
||||||
|
<li ng-show="hasPaidPlan" quay-require="['BILLING']">
|
||||||
|
<a href="javascript:void(0)" data-toggle="tab" data-target="#billing" ng-click="loadInvoices()">Billing History</a>
|
||||||
|
</li>
|
||||||
</ul>
|
</ul>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
@ -22,12 +30,12 @@
|
||||||
<div class="col-md-10">
|
<div class="col-md-10">
|
||||||
<div class="tab-content">
|
<div class="tab-content">
|
||||||
<!-- Plans tab -->
|
<!-- Plans tab -->
|
||||||
<div id="plan" class="tab-pane active">
|
<div id="plan" class="tab-pane active" quay-require="['BILLING']">
|
||||||
<div class="plan-manager" organization="orgname" plan-changed="planChanged(plan)"></div>
|
<div class="plan-manager" organization="orgname" plan-changed="planChanged(plan)"></div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<!-- Organization settings tab -->
|
<!-- Organization settings tab -->
|
||||||
<div id="settings" class="tab-pane">
|
<div id="settings" class="tab-pane" quay-classes="{'!BILLING': 'active'}">
|
||||||
<div class="quay-spinner" ng-show="changingOrganization"></div>
|
<div class="quay-spinner" ng-show="changingOrganization"></div>
|
||||||
|
|
||||||
<div class="panel" ng-show="!changingOrganization">
|
<div class="panel" ng-show="!changingOrganization">
|
||||||
|
@ -67,12 +75,12 @@
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<!-- Billing Options tab -->
|
<!-- Billing Options tab -->
|
||||||
<div id="billingoptions" class="tab-pane">
|
<div id="billingoptions" class="tab-pane" quay-require="['BILLING']">
|
||||||
<div class="billing-options" organization="organization"></div>
|
<div class="billing-options" organization="organization"></div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<!-- Billing History tab -->
|
<!-- Billing History tab -->
|
||||||
<div id="billing" class="tab-pane">
|
<div id="billing" class="tab-pane" quay-require="['BILLING']">
|
||||||
<div class="billing-invoices" organization="organization" visible="invoicesShown"></div>
|
<div class="billing-invoices" organization="organization" visible="invoicesShown"></div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|
|
@ -243,13 +243,14 @@
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<!-- Plans Table -->
|
<!-- Plans Table -->
|
||||||
<div class="form-group plan-group">
|
<div class="form-group plan-group" quay-require="['BILLING']">
|
||||||
<label>Organization Plan</label>
|
<label>Organization Plan</label>
|
||||||
<div class="plans-table" plans="orgPlans" current-plan="org.plan"></div>
|
<div class="plans-table" plans="orgPlans" current-plan="org.plan"></div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="button-bar">
|
<div class="button-bar">
|
||||||
<button class="btn btn-large btn-danger" type="submit" ng-disabled="convertForm.$invalid || !org.plan"
|
<button class="btn btn-large btn-danger" type="submit"
|
||||||
|
ng-disabled="convertForm.$invalid || (Features.BILLING && !org.plan)"
|
||||||
analytics-on analytics-event="convert_to_organization">
|
analytics-on analytics-event="convert_to_organization">
|
||||||
Convert To Organization
|
Convert To Organization
|
||||||
</button>
|
</button>
|
||||||
|
|
Reference in a new issue