Skip to content

Commit

Permalink
Merge pull request #1797 from openmeterio/feat/plan-api-impl
Browse files Browse the repository at this point in the history
refactor(api): Plan API
  • Loading branch information
tothandras authored Nov 7, 2024
2 parents b0c4884 + c2cb8c4 commit 3f4464b
Show file tree
Hide file tree
Showing 10 changed files with 1,895 additions and 2,206 deletions.
1,929 changes: 936 additions & 993 deletions api/api.gen.go

Large diffs are not rendered by default.

1,429 changes: 722 additions & 707 deletions api/client/go/client.gen.go

Large diffs are not rendered by default.

578 changes: 170 additions & 408 deletions api/openapi.yaml

Large diffs are not rendered by default.

71 changes: 8 additions & 63 deletions api/spec/src/productcatalog/discounts.tsp
Original file line number Diff line number Diff line change
Expand Up @@ -47,6 +47,14 @@ model DiscountPercentage {
@minValue(0)
@maxValue(100)
percentage: float;

/**
* The rate cards that the discount applies to.
* When not specified, the discount applies to all rate cards.
*/
@visibility("read", "create", "update")
@summary("Discounted RateCards")
rateCards?: Key[];
}

// NOTE(chrisgacsal): amount discount is going to be implemented in future releases
Expand Down Expand Up @@ -86,66 +94,3 @@ model DiscountPercentage {
// @summary("Usage")
// usage: float64;
// }

/**
* A discount on plan. One of: percentage or amount.
*/
@friendlyName("AppliedDiscount")
@discriminator("type")
union AppliedDiscount {
@summary("Percentage discount")
percentage: AppliedDiscountPercentage,

// @summary("Amount discount")
// amount: AppliedDiscountAmount,

// @summary("Usage discount")
// usage: AppliedDiscountUsage,
}

alias RateCardKey = Key;

/**
* Percentage discount on plan.
*/
@friendlyName("AppliedDiscountPercentage")
model AppliedDiscountPercentage {
...DiscountPercentage;

/**
* The rate cards that the discount applies to.
* When not specified, the discount applies to all rate cards.
*/
@summary("Applies To")
appliesTo?: RateCardKey[];
}

// /**
// * Percentage discount on plan.
// */
// @friendlyName("AppliedDiscountUsage")
// model AppliedDiscountUsage {
// ...DiscountUsage;

// /**
// * The rate cards that the discount applies to.
// * When not specified, the discount applies to all rate cards.
// */
// @summary("Applies To")
// appliesTo?: RateCardKey[];
// }

// /**
// * Amount discount on plan.
// */
// @friendlyName("AppliedDiscountAmount")
// model AppliedDiscountAmount {
// ...DiscountAmount;

// /**
// * The rate cards that the discount applies to.
// * When not specified, the discount applies to all rate cards.
// */
// @summary("Applies To")
// appliesTo?: RateCardKey[];
// }
14 changes: 11 additions & 3 deletions api/spec/src/productcatalog/plan.tsp
Original file line number Diff line number Diff line change
Expand Up @@ -52,18 +52,21 @@ model Plan {
/**
* The currency code of the plan.
*/
@visibility("read", "create")
@summary("Currency")
currency: CurrencyCode = "USD";

/**
* The date and time when the plan becomes effective. When not specified, the plan is a draft.
*/
@visibility("read")
@summary("Effective start date")
effectiveFrom?: DateTime;

/**
* The date and time when the plan is no longer effective. When not specified, the plan is effective indefinitely.
*/
@visibility("read")
@summary("Effective end date")
effectiveTo?: DateTime;

Expand All @@ -83,6 +86,7 @@ model Plan {
* The plan phase or pricing ramp allows changing a plan's rate cards over time as a subscription progresses.
* A phase switch occurs only at the end of a billing period, ensuring that a single subscription invoice will not include charges from different phase prices.
*/
@visibility("read", "create", "update")
@summary("Plan phases")
@minItems(1)
phases: PlanPhase[];
Expand All @@ -98,17 +102,20 @@ model PlanPhase {
/**
* Unique key among the plan phases. Used to reference the phase in the subscription & plan.
*/
@visibility("read", "create")
key: Key;

/**
* The rate cards of the plan.
*/
@visibility("read", "create", "update")
@summary("Rate cards")
rateCards: RateCard[];

/**
* The time after which the plan starts compared to subscription start
*/
@visibility("read", "create", "update")
@summary("Start after")
@encode(DurationKnownEncoding.ISO8601)
@example(duration.fromISO("P1Y1D"))
Expand All @@ -117,8 +124,9 @@ model PlanPhase {
/**
* The discounts on the plan.
*/
@visibility("read", "create", "update")
@summary("Discounts")
discounts?: AppliedDiscount[];
discounts?: Discount[];

// /**
// * Predefined overrides of the plan that can apply changes to the rate cards, discounts, and other properties based on the selected variant.
Expand Down Expand Up @@ -182,9 +190,9 @@ model PlanVariantOverridePreset {
};

/*
* The discount on the plan.
* The discounts on the plan.
*/
discount?: AppliedDiscount;
discounts?: Discount[];
}

/**
Expand Down
7 changes: 4 additions & 3 deletions api/spec/src/productcatalog/prices.tsp
Original file line number Diff line number Diff line change
Expand Up @@ -55,7 +55,7 @@ union PricePaymentTerm {
*/
@friendlyName("FlatPrice")
model FlatPrice {
@visibility("read", "create")
@visibility("read", "create", "update")
type: PriceType.flat;

/**
Expand All @@ -76,6 +76,7 @@ model FlatPriceWithPaymentTerm {
* The payment term of the flat price.
* Defaults to in advance.
*/
@visibility("read", "create", "update")
paymentTerm?: PricePaymentTerm = PricePaymentTerm.inAdvance;
}

Expand All @@ -84,7 +85,7 @@ model FlatPriceWithPaymentTerm {
*/
@friendlyName("UnitPrice")
model UnitPrice {
@visibility("read", "create")
@visibility("read", "create", "update")
type: PriceType.unit;

/**
Expand All @@ -108,7 +109,7 @@ enum TieredPriceMode {
*/
@friendlyName("TieredPrice")
model TieredPrice {
@visibility("read", "create")
@visibility("read", "create", "update")
type: PriceType.tiered;

/**
Expand Down
22 changes: 14 additions & 8 deletions api/spec/src/productcatalog/ratecards.tsp
Original file line number Diff line number Diff line change
Expand Up @@ -94,6 +94,18 @@ model RateCardFlatFee {
// discount?: DiscountPercentage | DiscountAmount;
}

/**
* The price of the usage based rate card.
*/
@friendlyName("RateCardUsageBasedPrice")
@discriminator("type")
@oneOf
union RateCardUsageBasedPrice {
flat: FlatPriceWithPaymentTerm,
unit: UnitPriceWithCommitments,
tiered: TieredPriceWithCommitments,
}

/**
* A usage-based rate card defines a price based on usage.
*/
Expand All @@ -109,18 +121,12 @@ model RateCardUsageBased {
@encode(DurationKnownEncoding.ISO8601)
billingCadence: duration;

/**
/*
* The price of the rate card.
* When null, the feature or service is free.
*/
@visibility("read", "create", "update")
@summary("Price")
@example(#{ type: PriceType.unit, amount: "0.01", minimumAmount: "100" })
price:
| UnitPriceWithCommitments
| TieredPriceWithCommitments
| FlatPriceWithPaymentTerm
| null;
price: RateCardUsageBasedPrice | null;

// NOTE(chrisgacsal): discount on RateCard level going to be implemented in upcoming releases
// /**
Expand Down
29 changes: 19 additions & 10 deletions api/spec/src/productcatalog/routes.tsp
Original file line number Diff line number Diff line change
Expand Up @@ -78,6 +78,15 @@ interface Plans {
@extension("x-go-type", "string")
@path
planId: PlanIdOrKey,

/**
* Include latest version of the Plan instead of the version in active state.
*
* Usage: `?includeLatest=true`
*/
@query
@example(true)
includeLatest?: boolean = false,
): Plan | OpenMeter.NotFoundError | OpenMeter.CommonErrors;

/**
Expand All @@ -93,7 +102,7 @@ interface Plans {
} | OpenMeter.NotFoundError | OpenMeter.CommonErrors;

/**
* Publish a new plan.
* Publish a plan version.
*/
@post
@route("/{planId}/publish")
Expand All @@ -105,13 +114,13 @@ interface Plans {
} | OpenMeter.NotFoundError | OpenMeter.CommonErrors;

/**
* Unpublish a new plan.
* Archive a plan version.
*/
@post
@route("/{planId}/unpublish")
@route("/{planId}/archive")
@sharedRoute
@operationId("unpublishPlan")
@summary("Unpublish plan")
@operationId("archivePlan")
@summary("Archive plan version")
unpublish(@path planId: ULID): {
@statusCode _: 204;
} | OpenMeter.NotFoundError | OpenMeter.CommonErrors;
Expand All @@ -123,7 +132,7 @@ interface Plans {
@post
@route("/{planIdOrKey}/next")
@sharedRoute
@operationId("newDraftPlan")
@operationId("nextPlan")
@summary("New draft plan")
next(
@extension("x-go-type", "string")
Expand Down Expand Up @@ -161,7 +170,7 @@ interface PlanPhases {
* Create new phase in plan.
*/
@post
@operationId("createPlanPhases")
@operationId("createPlanPhase")
@summary("Create new phase in plan")
create(
@path planId: ULID,
Expand All @@ -175,7 +184,7 @@ interface PlanPhases {
* Update phase in plan.
*/
@put
@operationId("updatePlanPhases")
@operationId("updatePlanPhase")
@summary("Update phase in plan")
update(
@path planId: ULID,
Expand All @@ -189,7 +198,7 @@ interface PlanPhases {
* Get phase in plan.
*/
@get
@operationId("getPlanPhases")
@operationId("getPlanPhase")
@summary("Get phase for plan")
get(
@path planId: ULID,
Expand All @@ -202,7 +211,7 @@ interface PlanPhases {
* Once a phase is deleted it cannot be undeleted.
*/
@delete
@operationId("deletePlanPhases")
@operationId("deletePlanPhase")
@summary("Delete phase for plan")
delete(@path planId: ULID, @path planPhaseKey: Key): {
@statusCode _: 204;
Expand Down
4 changes: 2 additions & 2 deletions api/spec/src/productcatalog/subscription.tsp
Original file line number Diff line number Diff line change
Expand Up @@ -105,8 +105,8 @@ model SubscriptionPhaseCreate {
/**
* The discounts on the plan.
*/
@summary("Discount")
discounts?: AppliedDiscount[];
@summary("Discounts")
discounts?: Discount[];
}

/**
Expand Down
18 changes: 9 additions & 9 deletions openmeter/server/router/plan.go
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,7 @@ func (a *Router) DeletePlan(w http.ResponseWriter, r *http.Request, planId strin

// Get plan
// (GET /api/v1/plans/{planId})
func (a *Router) GetPlan(w http.ResponseWriter, r *http.Request, planId string) {
func (a *Router) GetPlan(w http.ResponseWriter, r *http.Request, planId string, params api.GetPlanParams) {
w.WriteHeader(http.StatusNotImplemented)
}

Expand All @@ -38,7 +38,7 @@ func (a *Router) UpdatePlan(w http.ResponseWriter, r *http.Request, planId strin

// New draft plan
// (POST /api/v1/plans/{planId}/next)
func (a *Router) NewDraftPlan(w http.ResponseWriter, r *http.Request, planId string) {
func (a *Router) NextPlan(w http.ResponseWriter, r *http.Request, planId string) {
w.WriteHeader(http.StatusNotImplemented)
}

Expand All @@ -50,25 +50,25 @@ func (a *Router) ListPlanPhases(w http.ResponseWriter, r *http.Request, planId s

// Create new phase in plan
// (POST /api/v1/plans/{planId}/phases)
func (a *Router) CreatePlanPhases(w http.ResponseWriter, r *http.Request, planId string) {
func (a *Router) CreatePlanPhase(w http.ResponseWriter, r *http.Request, planId string) {
w.WriteHeader(http.StatusNotImplemented)
}

// Delete phase for plan
// (DELETE /api/v1/plans/{planId}/phases/{planPhaseKey})
func (a *Router) DeletePlanPhases(w http.ResponseWriter, r *http.Request, planId string, planPhaseKey string) {
func (a *Router) DeletePlanPhase(w http.ResponseWriter, r *http.Request, planId string, planPhaseKey string) {
w.WriteHeader(http.StatusNotImplemented)
}

// Get phase for plan
// (GET /api/v1/plans/{planId}/phases/{planPhaseKey})
func (a *Router) GetPlanPhases(w http.ResponseWriter, r *http.Request, planId string, planPhaseKey string) {
func (a *Router) GetPlanPhase(w http.ResponseWriter, r *http.Request, planId string, planPhaseKey string) {
w.WriteHeader(http.StatusNotImplemented)
}

// Update phase in plan
// (PUT /api/v1/plans/{planId}/phases/{planPhaseKey})
func (a *Router) UpdatePlanPhases(w http.ResponseWriter, r *http.Request, planId string, planPhaseKey string) {
func (a *Router) UpdatePlanPhase(w http.ResponseWriter, r *http.Request, planId string, planPhaseKey string) {
w.WriteHeader(http.StatusNotImplemented)
}

Expand All @@ -78,8 +78,8 @@ func (a *Router) PublishPlan(w http.ResponseWriter, r *http.Request, planId stri
w.WriteHeader(http.StatusNotImplemented)
}

// Unpublish plan
// (POST /api/v1/plans/{planId}/unpublish)
func (a *Router) UnpublishPlan(w http.ResponseWriter, r *http.Request, planId string) {
// Archive plan version
// (POST /api/v1/plans/{planId}/archive)
func (a *Router) ArchivePlan(w http.ResponseWriter, r *http.Request, planId string) {
w.WriteHeader(http.StatusNotImplemented)
}

0 comments on commit 3f4464b

Please sign in to comment.