UNPKG

smc-hub

Version:

CoCalc: Backend webserver component

266 lines 12.1 kB
"use strict"; /* * This file is part of CoCalc: Copyright © 2020 Sagemath, Inc. * License: AGPLv3 s.t. "Commons Clause" – see LICENSE.md for details */ var __assign = (this && this.__assign) || function () { __assign = Object.assign || function(t) { for (var s, i = 1, n = arguments.length; i < n; i++) { s = arguments[i]; for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p)) t[p] = s[p]; } return t; }; return __assign.apply(this, arguments); }; var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) { function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); } return new (P || (P = Promise))(function (resolve, reject) { function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } } function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } } function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); } step((generator = generator.apply(thisArg, _arguments || [])).next()); }); }; var __generator = (this && this.__generator) || function (thisArg, body) { var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g; return g = { next: verb(0), "throw": verb(1), "return": verb(2) }, typeof Symbol === "function" && (g[Symbol.iterator] = function() { return this; }), g; function verb(n) { return function (v) { return step([n, v]); }; } function step(op) { if (f) throw new TypeError("Generator is already executing."); while (_) try { if (f = 1, y && (t = op[0] & 2 ? y["return"] : op[0] ? y["throw"] || ((t = y["return"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t; if (y = 0, t) op = [op[0] & 2, t.value]; switch (op[0]) { case 0: case 1: t = op; break; case 4: _.label++; return { value: op[1], done: false }; case 5: _.label++; y = op[1]; op = [0]; continue; case 7: op = _.ops.pop(); _.trys.pop(); continue; default: if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; } if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; } if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; } if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; } if (t[2]) _.ops.pop(); _.trys.pop(); continue; } op = body.call(thisArg, _); } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; } if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true }; } }; var __values = (this && this.__values) || function(o) { var s = typeof Symbol === "function" && Symbol.iterator, m = s && o[s], i = 0; if (m) return m.call(o); if (o && typeof o.length === "number") return { next: function () { if (o && i >= o.length) o = void 0; return { value: o && o[i++], done: !o }; } }; throw new TypeError(s ? "Object is not iterable." : "Symbol.iterator is not defined."); }; Object.defineProperty(exports, "__esModule", { value: true }); exports.create_missing_plans = void 0; // STOPGAP FIX: relative dirs necessary for manage service var upgrade_spec_1 = require("smc-util/upgrade-spec"); var connection_1 = require("./connection"); // Create all plans that are missing function create_missing_plans(logger, database) { return __awaiter(this, void 0, void 0, function () { var dbg, stripe, plans, known, _a, _b, plan, _c, _d, _i, name_1; var e_1, _e; return __generator(this, function (_f) { switch (_f.label) { case 0: dbg = function (m) { return logger.debug("create_missing_plans: " + m); }; dbg(); // We have deprecated the old "upgrade" subscriptions, and the // code below would just ensure that all upgrade subscription types // described in code were also represented in stripe. However, // the existence of licenses which automatically created hundreds // of subscriptions broke this code (since only 100 plans get returned). // There's no point in fixing it now, since we won't be creating // any new subscriptions. dbg("DEPRECATED"); if (!false) return [3 /*break*/, 6]; dbg("initialize stripe connection"); return [4 /*yield*/, connection_1.init_stripe(database, logger)]; case 1: stripe = _f.sent(); dbg("get already created plans"); return [4 /*yield*/, stripe.plans.list({ limit: 999 })]; case 2: plans = _f.sent(); known = {}; try { for (_a = __values(plans.data), _b = _a.next(); !_b.done; _b = _a.next()) { plan = _b.value; known[plan.id] = true; } } catch (e_1_1) { e_1 = { error: e_1_1 }; } finally { try { if (_b && !_b.done && (_e = _a.return)) _e.call(_a); } finally { if (e_1) throw e_1.error; } } dbg("create any missing plans"); _c = []; for (_d in upgrade_spec_1.upgrades.subscription) _c.push(_d); _i = 0; _f.label = 3; case 3: if (!(_i < _c.length)) return [3 /*break*/, 6]; name_1 = _c[_i]; return [4 /*yield*/, create_plan(name_1, database, logger, known)]; case 4: _f.sent(); _f.label = 5; case 5: _i++; return [3 /*break*/, 3]; case 6: return [2 /*return*/]; } }); }); } exports.create_missing_plans = create_missing_plans; // Create a specific plan (error if plan already defined) function create_plan(name, // the name of the plan, one of the keys of upgrades.subscription; // NOTE: there are multiple stripe plans associated to a single cocalc // plan, due to different intervals. database, logger, known // map from known plan ids to true -- these are skipped ) { return __awaiter(this, void 0, void 0, function () { var spec, dbg, stripe, plans, plans_1, plans_1_1, plan, interval, e_2_1; var e_2, _a; return __generator(this, function (_b) { switch (_b.label) { case 0: spec = upgrade_spec_1.upgrades.subscription[name]; if (spec == null) { throw Error("unknown plan \"" + name + "\""); } dbg = function (m) { return logger.debug("create_plan(name=\"" + name + "\"): " + m); }; dbg(); dbg("initialize stripe connection"); return [4 /*yield*/, connection_1.init_stripe(database, logger)]; case 1: stripe = _b.sent(); plans = spec_to_plans(name, spec, known); if (plans.length === 0) { dbg("no missing stripe plans"); return [2 /*return*/]; } dbg("creating " + plans.length + " missing stripe plans"); _b.label = 2; case 2: _b.trys.push([2, 7, 8, 9]); plans_1 = __values(plans), plans_1_1 = plans_1.next(); _b.label = 3; case 3: if (!!plans_1_1.done) return [3 /*break*/, 6]; plan = plans_1_1.value; interval = plan.interval; if (interval != "day" && interval != "week" && interval != "month" && interval != "year") { // make TS happy, and a good consistency check -- this check above makes it so interval // has the right type, which is why we do the object merge below (to fix the typing). throw Error("invalid plan interval \"" + plan.interval + "\""); } return [4 /*yield*/, stripe.plans.create(__assign(__assign({}, plan), { interval: interval }))]; case 4: _b.sent(); _b.label = 5; case 5: plans_1_1 = plans_1.next(); return [3 /*break*/, 3]; case 6: return [3 /*break*/, 9]; case 7: e_2_1 = _b.sent(); e_2 = { error: e_2_1 }; return [3 /*break*/, 9]; case 8: try { if (plans_1_1 && !plans_1_1.done && (_a = plans_1.return)) _a.call(plans_1); } finally { if (e_2) throw e_2.error; } return [7 /*endfinally*/]; case 9: return [2 /*return*/]; } }); }); } function spec_to_plans(name, spec, known) { var v = []; var the_desc = spec.desc; var i = the_desc.indexOf("\n"); if (i !== -1) { the_desc = the_desc.slice(0, i); } for (var period in spec.price) { var desc = void 0, id = void 0, interval = void 0, interval_count = void 0; var amount = spec.price[period]; switch (period) { case "month": id = name; interval = "month"; interval_count = 1; desc = the_desc; break; case "month4": id = name; interval = "month"; interval_count = 4; desc = the_desc; break; case "year": case "year1": id = name + "-year"; interval = "year"; interval_count = 1; desc = "One Year " + the_desc; break; case "week": id = name + "-week"; interval = "week"; interval_count = 1; desc = "One Week " + the_desc; break; default: throw Error("unknown period '" + period + "'"); } if (known[id]) { continue; } var statement = spec.statement; if (statement == null) { throw Error("plan statement must be defined but it is not for name='" + name + "'"); } if (statement.length > 17) { throw Error("statement '" + statement + "' must be at most 17 characters, but is " + statement.length + " characters for name='" + name + "'"); } if (interval === "year") { statement += " YEAR"; } v.push({ id: id, interval: interval, interval_count: interval_count, amount: amount * 100, product: { name: desc, statement_descriptor: statement, }, currency: "usd", }); } return v; } //# sourceMappingURL=plans.js.map