DefaultPlanPhase.java

205 lines | 6.368 kB Blame History Raw Download
/*
 * Copyright 2010-2011 Ning, Inc.
 *
 * Ning licenses this file to you under the Apache License, version 2.0
 * (the "License"); you may not use this file except in compliance with the
 * License.  You may obtain a copy of the License at:
 *
 *    http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
 * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.  See the
 * License for the specific language governing permissions and limitations
 * under the License.
 */

package com.ning.billing.catalog;

import com.ning.billing.ErrorCode;
import com.ning.billing.catalog.api.BillingPeriod;
import com.ning.billing.catalog.api.CatalogApiException;
import com.ning.billing.catalog.api.Duration;
import com.ning.billing.catalog.api.InternationalPrice;
import com.ning.billing.catalog.api.PhaseType;
import com.ning.billing.catalog.api.Plan;
import com.ning.billing.catalog.api.PlanPhase;
import com.ning.billing.util.config.ValidatingConfig;
import com.ning.billing.util.config.ValidationError;
import com.ning.billing.util.config.ValidationErrors;

import javax.xml.bind.annotation.XmlAccessType;
import javax.xml.bind.annotation.XmlAccessorType;
import javax.xml.bind.annotation.XmlAttribute;
import javax.xml.bind.annotation.XmlElement;
import java.net.URI;

@XmlAccessorType(XmlAccessType.NONE)
public class DefaultPlanPhase extends ValidatingConfig<StandaloneCatalog> implements PlanPhase {

	@XmlAttribute (required=true)
	private PhaseType type;

    @XmlElement(required=true)
    private DefaultDuration duration;
    
    @XmlElement(required=true)
    private BillingPeriod billingPeriod;

	@XmlElement(required=false)
	private DefaultInternationalPrice recurringPrice;

	@XmlElement(required=false)
	private DefaultInternationalPrice fixedPrice;

//  Not supported: variable pricing
//	@XmlElement(required=false)
//	private InternationalPrice unitPrice;

	//Not exposed in XML
	private Plan plan;
	
	public static String phaseName(Plan plan, PlanPhase phase) {
		return plan.getName() + "-" + phase.getPhaseType().toString().toLowerCase();
	}
	
	public static String planName(String phaseName) throws CatalogApiException {
		for(PhaseType type : PhaseType.values()) {
			if(phaseName.endsWith(type.toString().toLowerCase())) {
				return phaseName.substring(0, phaseName.length() - type.toString().length() - 1);
			}
		}
		throw new CatalogApiException(ErrorCode.CAT_BAD_PHASE_NAME, phaseName);
	}


	/* (non-Javadoc)
	 * @see com.ning.billing.catalog.IPlanPhase#getRecurringPrice()
	 */
    @Override
	public DefaultInternationalPrice getRecurringPrice() {
        return recurringPrice;
    }

	/* (non-Javadoc)
	 * @see com.ning.billing.catalog.IPlanPhase#getInternationalPrice()
	 */
    @Override
	public InternationalPrice getFixedPrice() {
        return fixedPrice;
    }

	/* (non-Javadoc)
	 * @see com.ning.billing.catalog.IPlanPhase#getCohort()
	 */
	@Override
	public PhaseType getPhaseType() {
		return type;
	}

	/* (non-Javadoc)
	 * @see com.ning.billing.catalog.IPlanPhase#getBillCycleDuration()
	 */
    @Override
	public BillingPeriod getBillingPeriod() {
    	return billingPeriod;
    }

	/* (non-Javadoc)
	 * @see com.ning.billing.catalog.IPlanPhase#getName()
	 */
	@Override
	public String getName() {
		return phaseName(plan,this);
	}

	/* (non-Javadoc)
	 * @see com.ning.billing.catalog.IPlanPhase#getPlan()
	 */
	@Override
	public Plan getPlan() {
		return plan;
	}

	/* (non-Javadoc)
	 * @see com.ning.billing.catalog.IPlanPhase#getDuration()
	 */
	@Override
	public Duration getDuration() {
 		return duration;
	}

	@Override
	public ValidationErrors validate(StandaloneCatalog catalog, ValidationErrors errors) {
		//Validation: check for nulls
		if(billingPeriod == null) {
			errors.add(new ValidationError(String.format("Phase %s of plan %s has a recurring price but no billing period", type.toString(), plan.getName()),
					catalog.getCatalogURI(), DefaultPlanPhase.class, type.toString()));
		}
	
		//Validation: if there is a recurring price there must be a billing period
		if((recurringPrice != null) && (billingPeriod == null || billingPeriod == BillingPeriod.NO_BILLING_PERIOD)) {
			errors.add(new ValidationError(String.format("Phase %s of plan %s has a recurring price but no billing period", type.toString(), plan.getName()),
					catalog.getCatalogURI(), DefaultPlanPhase.class, type.toString()));
		}

		//Validation: if there is no recurring price there should be no billing period
		if((recurringPrice == null) && billingPeriod != BillingPeriod.NO_BILLING_PERIOD) {
			errors.add(new ValidationError(String.format("Phase %s of plan %s has no recurring price but does have a billing period. The billing period should be set to '%s'",
					type.toString(), plan.getName(), BillingPeriod.NO_BILLING_PERIOD), 
					catalog.getCatalogURI(), DefaultPlanPhase.class, type.toString()));
		}
		
		//Validation: there must be at least one of recurringPrice or fixedPrice
		if((recurringPrice == null) && fixedPrice == null) {
			errors.add(new ValidationError(String.format("Phase %s of plan %s has neither a recurring price or a fixed price.",
					type.toString(), plan.getName()), 
					catalog.getCatalogURI(), DefaultPlanPhase.class, type.toString()));
		}
		//TODO : if there BP is set to NO_BILLING_PERIOD there must be a recurring price
        return errors;
	}
	
	@Override
	public void initialize(StandaloneCatalog root, URI uri) {
		if (fixedPrice != null) { fixedPrice.initialize(root, uri);  }	
		if (recurringPrice != null) { recurringPrice.initialize(root, uri); }
	}
	
	protected DefaultPlanPhase setFixedPrice(DefaultInternationalPrice price) {
		this.fixedPrice = price;
		return this;
	}

	protected DefaultPlanPhase setRecurringPrice(DefaultInternationalPrice price) {
		this.recurringPrice = price;
		return this;
	}

	protected DefaultPlanPhase setPhaseType(PhaseType cohort) {
		this.type = cohort;
		return this;
	}

	protected DefaultPlanPhase setBillingPeriod(BillingPeriod billingPeriod) {
		this.billingPeriod = billingPeriod;
		return this;
	}

	protected DefaultPlanPhase setDuration(DefaultDuration duration) {
		this.duration = duration;
		return this;
	}

	protected DefaultPlanPhase setPlan(Plan plan) {
		this.plan = plan;
		return this;
	}
	
	protected DefaultPlanPhase setBillCycleDuration(BillingPeriod billingPeriod) {
		this.billingPeriod = billingPeriod;
		return this;
	}

}