From 3097a5539efc0469850ea9863c4eb75982ffa0be Mon Sep 17 00:00:00 2001
From: Bart Arendarczyk <s1924442@ed.ac.uk>
Date: Fri, 2 Jun 2023 17:36:40 +0100
Subject: [PATCH] Various carbon market changes and fixes.

---
 GAMS/IntExtOpt.gms                            |  4 +-
 src/ac/ed/lurg/ModelConfig.java               |  4 +-
 src/ac/ed/lurg/ModelMain.java                 | 14 +++----
 src/ac/ed/lurg/carbon/CarbonFluxItem.java     | 22 +++++++---
 src/ac/ed/lurg/carbon/CarbonFluxReader.java   | 42 ++++---------------
 src/ac/ed/lurg/country/GlobalPrice.java       |  4 +-
 .../country/gams/GamsLocationOptimiser.java   | 18 ++++++--
 .../lurg/demand/AbstractSSPDemandManager.java |  2 +-
 .../ed/lurg/landuse/ConversionCostReader.java |  3 +-
 9 files changed, 56 insertions(+), 57 deletions(-)

diff --git a/GAMS/IntExtOpt.gms b/GAMS/IntExtOpt.gms
index 9bb90990..2a0a6d8f 100644
--- a/GAMS/IntExtOpt.gms
+++ b/GAMS/IntExtOpt.gms
@@ -50,6 +50,7 @@
  PARAMETER minDemandFraction(commodity, all_types)   min demand for each cereal crop as factor of all cereals;
  PARAMETER seedAndWasteRate(all_types)       rate of use for seed and waste combined;
  PARAMETER subsidyRate(all_types)                 rates of subsidy compared to costs;
+ PARAMETER tradeAdjustmentCostRate(import_types)  cost of adjusting net imports;
  
  PARAMETER previousLandCoverArea(land_cover, location)                      land cover area in Mha;
  PARAMETER maxCroplandArea(location);
@@ -70,7 +71,6 @@
  SCALAR maxGrossLccRate                      max rate of gross land cover change;
  SCALAR maxFertChange                        max increase in total fertilizer use;
  SCALAR maxIrrigChange                       max increase in total irrigation use;
- SCALAR tradeAdjustmentCostRate;
 
  SCALAR forestManagementCost    cost $1000 per ha;
  
@@ -332,7 +332,7 @@ $gdxin
  MAX_NET_IMPORT_CONSTRAINT(import_types) .. netImportAmount(import_types) =L= maxNetImport(import_types);
  MIN_NET_IMPORT_CONSTRAINT(import_types) .. netImportAmount(import_types) =G= minNetImport(import_types);
  
- TRADE_ADJUSTMENT_COST_CALC(import_types) .. tradeAdjustmentCost(import_types) =E= power(netImportAmount(import_types) - previousNetImport(import_types), 2) * tradeAdjustmentCostRate;
+ TRADE_ADJUSTMENT_COST_CALC(import_types) .. tradeAdjustmentCost(import_types) =E= power(netImportAmount(import_types) - previousNetImport(import_types), 2) * tradeAdjustmentCostRate(import_types);
  
 * No imports or exports of pasture allowed
  importAmount.FX('pasture') = 0;
diff --git a/src/ac/ed/lurg/ModelConfig.java b/src/ac/ed/lurg/ModelConfig.java
index 3b1a7379..bb165cc0 100755
--- a/src/ac/ed/lurg/ModelConfig.java
+++ b/src/ac/ed/lurg/ModelConfig.java
@@ -597,6 +597,8 @@ public class ModelConfig {
 	public static final double INIT_CARBON_STOCK = getDoubleProperty("INIT_CARBON_STOCK", 100.0); // MtC-eq
 	public static final String CARBON_OPTIONS_FILENAME = getProperty("CARBON_OPTIONS_FILENAME", "carbon_options.csv"); // config for what is included in carbon market
 	public static final String CARBON_OPTIONS_FILE = getProperty("CARBON_OPTIONS_FILE", DATA_DIR + File.separator + CARBON_OPTIONS_FILENAME);
-	public static final double CARBON_FOREST_MAX_PROPORTION = getDoubleProperty("CARBON_FOREST_MAX_PROPORTION", 0.2); // maximum proportion of land cover as carbon forest
+	public static final double CARBON_FOREST_MAX_PROPORTION = getDoubleProperty("CARBON_FOREST_MAX_PROPORTION", 1.0); // maximum proportion of land cover as carbon forest
+	public static final double CARBON_FOREST_CONVERSION_PENALTY = getDoubleProperty("CARBON_FOREST_CONVERSION_PENALTY", 10.0); // Penality for removing carbon forest 1000$/ha
+	public static final double CARBON_TRADE_ADJ_COST_MULTIPLIER = getDoubleProperty("CARBON_TRADE_ADJ_COST_MULTIPLIER", 0.1); // No physical goods traded so can have lower cost
 
 }
diff --git a/src/ac/ed/lurg/ModelMain.java b/src/ac/ed/lurg/ModelMain.java
index c3871411..72d76eec 100644
--- a/src/ac/ed/lurg/ModelMain.java
+++ b/src/ac/ed/lurg/ModelMain.java
@@ -637,16 +637,12 @@ public class ModelMain {
 	}
 	
 	private Map<CompositeCountry, CarbonUsageData> getInitialCarbonUsageData() {
-		Map<CompositeCountry, CarbonUsageData> carbonUsageDataMap;
-		if (ModelConfig.IS_CALIBRATION_RUN) {
-			carbonUsageDataMap = new HashMap<CompositeCountry, CarbonUsageData>();
-			for (CompositeCountry cc : CountryManager.getInstance().getAllCompositeCountries()) {
-				CarbonUsageData cuData = new CarbonUsageData(0, 0, 0);
-				carbonUsageDataMap.put(cc, cuData);
-			}
-		} else {
-			carbonUsageDataMap = deserializeCarbonUsage();
+		Map<CompositeCountry, CarbonUsageData> carbonUsageDataMap = new HashMap<>();
+		for (CompositeCountry cc : CountryManager.getInstance().getAllCompositeCountries()) {
+			CarbonUsageData cuData = new CarbonUsageData(0, 0, 0);
+			carbonUsageDataMap.put(cc, cuData);
 		}
+
 		return carbonUsageDataMap;
 	}
 
diff --git a/src/ac/ed/lurg/carbon/CarbonFluxItem.java b/src/ac/ed/lurg/carbon/CarbonFluxItem.java
index f8602297..a6f2210e 100644
--- a/src/ac/ed/lurg/carbon/CarbonFluxItem.java
+++ b/src/ac/ed/lurg/carbon/CarbonFluxItem.java
@@ -15,14 +15,27 @@ import ac.ed.lurg.types.LandProtectionType;
 public class CarbonFluxItem implements RasterItem, Serializable {
 	private static final long serialVersionUID = 440720456140537815L;
 	Map<LccKey, Double> carbonCreditRate = new HashMap<LccKey, Double>();
-
 	
-	public void calcCarbonCreditRate(Double[] cFluxes) {
+	public void calcCarbonCreditRate(Double[] cFluxes, LandCoverType lcType) {
+		double totalFlux = 0;
+		for (int i = 0; i <= ModelConfig.CARBON_HORIZON; i++) {
+			totalFlux += cFluxes[i];
+		}
 
-	}
+		totalFlux *= -1; // flip sign so positive mean net carbon uptake
 
+		// Tallying up different in net carbon uptake for each LCC
+		// so that carbonCreditRate(A -> B) = net C uptake for land cover b - net C uptake for land cover A
+		for (LandCoverType otherLc : LandCoverType.getConvertibleTypes()) {
+			if (otherLc.equals(lcType))
+				continue;
+			// Converting to this type so add to carbon credit rate
+			carbonCreditRate.merge(new LccKey(otherLc, lcType), totalFlux, Double::sum);
+			// Converting from this type so subtract from carbon credit rate
+			//carbonCreditRate.merge(new LccKey(lcType, otherLc), -totalFlux, Double::sum);
+		}
+	}
 
-	
 	public void setCarbonCreditRate(LccKey lcType, double cFlux) {
 		carbonCreditRate.put(lcType, cFlux);
 	}
@@ -31,7 +44,6 @@ public class CarbonFluxItem implements RasterItem, Serializable {
 		carbonCreditRate.put(LccKey.createKey(fromLc, toLc), cFlux);
 	}
 
-	
 	public double getCarbonCreditRate(LandCoverType fromLc, LandCoverType toLc) {
 		return carbonCreditRate.getOrDefault(LccKey.createKey(fromLc, toLc), Double.NaN);
 	}
diff --git a/src/ac/ed/lurg/carbon/CarbonFluxReader.java b/src/ac/ed/lurg/carbon/CarbonFluxReader.java
index 99f89a6b..c2367e79 100644
--- a/src/ac/ed/lurg/carbon/CarbonFluxReader.java
+++ b/src/ac/ed/lurg/carbon/CarbonFluxReader.java
@@ -13,7 +13,6 @@ import java.util.Map;
 import ac.ed.lurg.ModelConfig;
 import ac.ed.lurg.Timestep;
 import ac.ed.lurg.landuse.LandUseItem;
-import ac.ed.lurg.landuse.LccKey;
 import ac.ed.lurg.types.LandCoverType;
 import ac.ed.lurg.utils.LogWriter;
 import ac.sac.raster.RasterHeaderDetails;
@@ -35,44 +34,19 @@ public class CarbonFluxReader {
 		keyMap = getRasterKeys(cFluxData);
 		modelYear = ModelConfig.IS_CALIBRATION_RUN ? ModelConfig.BASE_YEAR : timestep.getYear();
 		long startTime = System.currentTimeMillis();
-		
-		readLccData(ModelConfig.C_FLUX_NTRL_TO_PAST_FILENAME, cFluxData, landUseRaster, new LccKey(LandCoverType.NATURAL, LandCoverType.PASTURE), false);
-		readLccData(ModelConfig.C_FLUX_NTRL_TO_FORS_FILENAME, cFluxData, landUseRaster,  new LccKey(LandCoverType.NATURAL, LandCoverType.CARBON_FOREST), false);
-		readLccData(ModelConfig.C_FLUX_NTRL_TO_FORS_FILENAME, cFluxData, landUseRaster,  new LccKey(LandCoverType.NATURAL, LandCoverType.TIMBER_FOREST), false);
-		readLccData(ModelConfig.C_FLUX_NTRL_TO_CROP_FILENAME, cFluxData, landUseRaster,  new LccKey(LandCoverType.NATURAL, LandCoverType.CROPLAND), false);
-		
-		readLccData(ModelConfig.C_FLUX_PAST_TO_NTRL_FILENAME, cFluxData, landUseRaster,  new LccKey(LandCoverType.PASTURE, LandCoverType.NATURAL), false);
-		readLccData(ModelConfig.C_FLUX_PAST_TO_FORS_FILENAME, cFluxData, landUseRaster,  new LccKey(LandCoverType.PASTURE, LandCoverType.CARBON_FOREST), false);
-		readLccData(ModelConfig.C_FLUX_PAST_TO_FORS_FILENAME, cFluxData, landUseRaster,  new LccKey(LandCoverType.PASTURE, LandCoverType.TIMBER_FOREST), false);
-		readLccData(ModelConfig.C_FLUX_PAST_TO_CROP_FILENAME, cFluxData, landUseRaster,  new LccKey(LandCoverType.PASTURE, LandCoverType.CROPLAND), false);
-		
-		readLccData(ModelConfig.C_FLUX_FORS_TO_NTRL_FILENAME, cFluxData, landUseRaster,  new LccKey(LandCoverType.CARBON_FOREST, LandCoverType.NATURAL), false);
-		readLccData(ModelConfig.C_FLUX_FORS_TO_PAST_FILENAME, cFluxData, landUseRaster,  new LccKey(LandCoverType.CARBON_FOREST, LandCoverType.PASTURE), false);
-		readLccData(ModelConfig.C_FLUX_FORS_TO_FORS_FILENAME, cFluxData, landUseRaster,  new LccKey(LandCoverType.CARBON_FOREST, LandCoverType.TIMBER_FOREST), false);
-		readLccData(ModelConfig.C_FLUX_FORS_TO_CROP_FILENAME, cFluxData, landUseRaster,  new LccKey(LandCoverType.CARBON_FOREST, LandCoverType.CROPLAND), false);
-		
-		readLccData(ModelConfig.C_FLUX_FORS_TO_NTRL_FILENAME, cFluxData, landUseRaster,  new LccKey(LandCoverType.TIMBER_FOREST, LandCoverType.NATURAL), false);
-		readLccData(ModelConfig.C_FLUX_FORS_TO_PAST_FILENAME, cFluxData, landUseRaster,  new LccKey(LandCoverType.TIMBER_FOREST, LandCoverType.PASTURE), false);
-		readLccData(ModelConfig.C_FLUX_FORS_TO_FORS_FILENAME, cFluxData, landUseRaster,  new LccKey(LandCoverType.TIMBER_FOREST, LandCoverType.CARBON_FOREST), false);
-		readLccData(ModelConfig.C_FLUX_FORS_TO_CROP_FILENAME, cFluxData, landUseRaster,  new LccKey(LandCoverType.TIMBER_FOREST, LandCoverType.CROPLAND), false);
-		
-		readLccData(ModelConfig.C_FLUX_CROP_TO_NTRL_FILENAME, cFluxData, landUseRaster,  new LccKey(LandCoverType.CROPLAND, LandCoverType.NATURAL), false);
-		readLccData(ModelConfig.C_FLUX_CROP_TO_PAST_FILENAME, cFluxData, landUseRaster,  new LccKey(LandCoverType.CROPLAND, LandCoverType.PASTURE), false);
-		readLccData(ModelConfig.C_FLUX_CROP_TO_FORS_FILENAME, cFluxData, landUseRaster,  new LccKey(LandCoverType.CROPLAND, LandCoverType.CARBON_FOREST), false);
-		readLccData(ModelConfig.C_FLUX_CROP_TO_FORS_FILENAME, cFluxData, landUseRaster,  new LccKey(LandCoverType.CROPLAND, LandCoverType.TIMBER_FOREST), false);
-		
-		readLccData(ModelConfig.C_FLUX_NTRL_FILENAME, cFluxData, landUseRaster, new LccKey(LandCoverType.NATURAL, LandCoverType.NATURAL), true);
-		readLccData(ModelConfig.C_FLUX_PAST_FILENAME, cFluxData, landUseRaster, new LccKey(LandCoverType.PASTURE, LandCoverType.PASTURE), true);
-		readLccData(ModelConfig.C_FLUX_FORS_FILENAME, cFluxData, landUseRaster, new LccKey(LandCoverType.CARBON_FOREST, LandCoverType.CARBON_FOREST), true);
-		readLccData(ModelConfig.C_FLUX_FORS_FILENAME, cFluxData, landUseRaster, new LccKey(LandCoverType.TIMBER_FOREST, LandCoverType.TIMBER_FOREST), true);
-		readLccData(ModelConfig.C_FLUX_CROP_FILENAME, cFluxData, landUseRaster, new LccKey(LandCoverType.CROPLAND, LandCoverType.CROPLAND), true);
+
+		//readData(ModelConfig.C_FLUX_NTRL_FILENAME, cFluxData, landUseRaster, LandCoverType.NATURAL);
+		//readData(ModelConfig.C_FLUX_PAST_FILENAME, cFluxData, landUseRaster, LandCoverType.PASTURE);
+		readData(ModelConfig.C_FLUX_FORS_FILENAME, cFluxData, landUseRaster, LandCoverType.CARBON_FOREST);
+		//readData(ModelConfig.C_FLUX_FORS_FILENAME, cFluxData, landUseRaster, LandCoverType.TIMBER_FOREST);
+		//readData(ModelConfig.C_FLUX_CROP_FILENAME, cFluxData, landUseRaster, LandCoverType.CROPLAND);
 		
 		LogWriter.println("Reading carbon data took " + (System.currentTimeMillis() - startTime) + " ms");
 		return cFluxData;
 
 	}
 	
-	public void readLccData(String fileName, CarbonFluxRasterSet carbonFluxData, RasterSet<LandUseItem> landUseRaster, LccKey lccKey, boolean neeCalc) {
+	public void readData(String fileName, CarbonFluxRasterSet carbonFluxData, RasterSet<LandUseItem> landUseRaster, LandCoverType lcType) {
 		long startTime = System.currentTimeMillis();
 		int maxYear = Math.floorDiv(modelYear - 10, ModelConfig.CARBON_DATA_TIMESTEP_SIZE) * ModelConfig.CARBON_DATA_TIMESTEP_SIZE + 10;
 		int maxAge = modelYear - ModelConfig.CARBON_DATA_MIN_YEAR;
@@ -108,7 +82,7 @@ public class CarbonFluxReader {
 			if (landUseRaster.containsKey(key)) {
 				CarbonFluxItem cfItem = carbonFluxData.get(key.getCol(), key.getRow());
 				LandUseItem luItem = landUseRaster.get(key);
-				cfItem.calcCarbonCreditRate(map.get(key));
+				cfItem.calcCarbonCreditRate(map.get(key), lcType);
 			}
 		}
 		LogWriter.println("Reading " + fileName + " took " + (System.currentTimeMillis() - startTime) + " ms");
diff --git a/src/ac/ed/lurg/country/GlobalPrice.java b/src/ac/ed/lurg/country/GlobalPrice.java
index 94b83d63..552b10b7 100644
--- a/src/ac/ed/lurg/country/GlobalPrice.java
+++ b/src/ac/ed/lurg/country/GlobalPrice.java
@@ -191,7 +191,9 @@ public class GlobalPrice implements Serializable {
 	}
 
 	public void setStockToTargetLevel(double targetStockUseRatio) {
-		stockLevel = production * targetStockUseRatio;
+		if (!Double.isNaN(production)) {
+			stockLevel = production * targetStockUseRatio;
+		}
 	}
 
 	public double getTradeChangeUp(double maxOfProdOrSupply) {
diff --git a/src/ac/ed/lurg/country/gams/GamsLocationOptimiser.java b/src/ac/ed/lurg/country/gams/GamsLocationOptimiser.java
index b56b4591..3603d4a5 100644
--- a/src/ac/ed/lurg/country/gams/GamsLocationOptimiser.java
+++ b/src/ac/ed/lurg/country/gams/GamsLocationOptimiser.java
@@ -380,12 +380,24 @@ public class GamsLocationOptimiser {
 		addScalar(inDB, "maxIrrigChange", (ModelConfig.IS_CALIBRATION_RUN
 				&& inputData.getTimestep().isInitialTimestep()) ? 1 : ModelConfig.MAX_IRRIG_CHANGE, 6);
 
+		// Setting the same trade adjustment cost for all commodities apart from carbonCredits.
+		// Could have individual values if needed
+
+		GAMSParameter tradeAdjCostP = inDB.addParameter("tradeAdjustmentCostRate", 1);
 		double tradeAdjCost = ModelConfig.getAdjParam("TRADE_ADJUSTMENT_COST_RATE");
-		addScalar(inDB, "tradeAdjustmentCostRate", tradeAdjCost, 6);
 
-		// Forestry
-		GAMSParameter maxRotaIntensityP = inDB.addParameter("maxRotationIntensity", 1);
+		for (CropType crop : CropType.getImportedTypes()) {
+			setGamsParamValue(tradeAdjCostP.addRecord(crop.getGamsName()), tradeAdjCost, -1);
+		}
 
+		for (WoodType wType : WoodType.values()) {
+			setGamsParamValue(tradeAdjCostP.addRecord(wType.getName()), tradeAdjCost, -1);
+		}
+
+		setGamsParamValue(tradeAdjCostP.addRecord("carbonCredits"),
+				ModelConfig.CARBON_TRADE_ADJ_COST_MULTIPLIER * tradeAdjCost, -1);
+
+		// Forestry
 		Map<WoodType, TradeConstraint> woodTradeConstraints = countryInput.getWoodTradeConstraints();
 		Map<WoodType, CountryPrice> woodPrices = countryInput.getWoodPrices();
 		Map<WoodType, Double> woodDemandMap = countryInput.getWoodDemand();
diff --git a/src/ac/ed/lurg/demand/AbstractSSPDemandManager.java b/src/ac/ed/lurg/demand/AbstractSSPDemandManager.java
index 87e6fe88..2be319e7 100644
--- a/src/ac/ed/lurg/demand/AbstractSSPDemandManager.java
+++ b/src/ac/ed/lurg/demand/AbstractSSPDemandManager.java
@@ -113,7 +113,7 @@ public abstract class AbstractSSPDemandManager extends AbstractDemandManager {
 	
 	protected double getCarbonDemand(SingleCountry country, int year) {
 		
-		if (!ModelConfig.IS_CARBON_ON)
+		if (!ModelConfig.IS_CARBON_ON | ModelConfig.IS_CALIBRATION_RUN)
 			return 0;
 		
 		SspData sd = sspManager.get(ssp_scenario, year, country);
diff --git a/src/ac/ed/lurg/landuse/ConversionCostReader.java b/src/ac/ed/lurg/landuse/ConversionCostReader.java
index 8779d240..7512a019 100644
--- a/src/ac/ed/lurg/landuse/ConversionCostReader.java
+++ b/src/ac/ed/lurg/landuse/ConversionCostReader.java
@@ -80,7 +80,8 @@ public class ConversionCostReader {
 					conversionCosts.put(key, ModelConfig.getAdjParam("FOREST_CONVERSION_COST") * costAdjFactor);
 					break;
 				case CARBON_FOREST:
-					conversionCosts.put(key, ModelConfig.getAdjParam("FOREST_CONVERSION_COST") * costAdjFactor);
+					conversionCosts.put(key, ModelConfig.getAdjParam("FOREST_CONVERSION_COST") * costAdjFactor
+							+ ModelConfig.CARBON_FOREST_CONVERSION_PENALTY);
 					break;
 				case NATURAL:
 					conversionCosts.put(key, ModelConfig.getAdjParam("NATURAL_CONVERSION_COST") * costAdjFactor);
-- 
GitLab