Compare commits
No commits in common. "0d99ee219da87ca872a82982082d711fd11553c3" and "99e8f305e1b5b519fc08bc522cd261f2eeb48558" have entirely different histories.
0d99ee219d
...
99e8f305e1
|
@ -1,7 +0,0 @@
|
||||||
package ch.zhaw.gartenverwaltung.io;
|
|
||||||
|
|
||||||
public class HardinessZoneNotSetException extends Exception {
|
|
||||||
public HardinessZoneNotSetException() {
|
|
||||||
super("HardinessZone must be set to retrieve plants!");
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -11,25 +11,12 @@ import java.net.URL;
|
||||||
import java.time.MonthDay;
|
import java.time.MonthDay;
|
||||||
import java.time.format.DateTimeFormatter;
|
import java.time.format.DateTimeFormatter;
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
import java.util.HashMap;
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
|
||||||
import java.util.Optional;
|
import java.util.Optional;
|
||||||
|
|
||||||
/**
|
|
||||||
* Implements the {@link PlantDatabase} interface for loading {@link Plant} objects
|
|
||||||
* from a JSON file.
|
|
||||||
* The reads are cached to minimize file-io operations.
|
|
||||||
*/
|
|
||||||
public class JsonPlantDatabase implements PlantDatabase {
|
public class JsonPlantDatabase implements PlantDatabase {
|
||||||
private final URL dataSource = getClass().getResource("plantdb.json");
|
private final URL dataSource = getClass().getResource("plantdb.json");
|
||||||
|
|
||||||
private HardinessZone currentZone;
|
|
||||||
private Map<Long, Plant> plantMap = Collections.emptyMap();
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Creating constant objects required to deserialize the {@link MonthDay} classes
|
|
||||||
*/
|
|
||||||
private final static JavaTimeModule timeModule = new JavaTimeModule();
|
private final static JavaTimeModule timeModule = new JavaTimeModule();
|
||||||
static {
|
static {
|
||||||
DateTimeFormatter dateFormat = DateTimeFormatter.ofPattern("MM-dd");
|
DateTimeFormatter dateFormat = DateTimeFormatter.ofPattern("MM-dd");
|
||||||
|
@ -37,59 +24,22 @@ public class JsonPlantDatabase implements PlantDatabase {
|
||||||
timeModule.addDeserializer(MonthDay.class, dateDeserializer);
|
timeModule.addDeserializer(MonthDay.class, dateDeserializer);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* If no data is currently loaded, or the specified zone differs
|
|
||||||
* from the {@link #currentZone}, data is loaded from {@link #dataSource}.
|
|
||||||
* In any case, the values of {@link #plantMap} are returned.
|
|
||||||
*
|
|
||||||
* @see PlantDatabase#getPlantList(HardinessZone)
|
|
||||||
*/
|
|
||||||
@Override
|
@Override
|
||||||
public List<Plant> getPlantList(HardinessZone zone) throws IOException, HardinessZoneNotSetException {
|
public List<Plant> getPlantList(HardinessZone zone) throws IOException {
|
||||||
if (plantMap.isEmpty() || zone != currentZone) {
|
List<Plant> result = Collections.emptyList();
|
||||||
loadPlantList(zone);
|
|
||||||
}
|
|
||||||
return plantMap.values().stream().toList();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @see PlantDatabase#getPlantById(long)
|
|
||||||
*/
|
|
||||||
@Override
|
|
||||||
public Optional<Plant> getPlantById(long id) throws HardinessZoneNotSetException, IOException {
|
|
||||||
if (plantMap.isEmpty()) {
|
|
||||||
loadPlantList(currentZone);
|
|
||||||
}
|
|
||||||
return Optional.ofNullable(plantMap.get(id));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Loads the database from {@link #dataSource} and updates the cached data.
|
|
||||||
*
|
|
||||||
* @param zone The {@link HardinessZone} for which data is to be loaded
|
|
||||||
* @throws IOException If the database cannot be accessed
|
|
||||||
*/
|
|
||||||
private void loadPlantList(HardinessZone zone) throws IOException, HardinessZoneNotSetException {
|
|
||||||
if (zone == null) {
|
|
||||||
throw new HardinessZoneNotSetException();
|
|
||||||
}
|
|
||||||
if (dataSource != null) {
|
if (dataSource != null) {
|
||||||
currentZone = zone;
|
|
||||||
ObjectMapper mapper = new ObjectMapper();
|
ObjectMapper mapper = new ObjectMapper();
|
||||||
mapper.registerModule(timeModule);
|
mapper.registerModule(timeModule);
|
||||||
|
|
||||||
List<Plant> result;
|
|
||||||
result = mapper.readerForListOf(Plant.class).readValue(dataSource);
|
result = mapper.readerForListOf(Plant.class).readValue(dataSource);
|
||||||
|
|
||||||
for (Plant plant : result) {
|
|
||||||
plant.inZone(currentZone);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Turn list into a HashMap with structure id => Plant
|
return result;
|
||||||
plantMap = result.stream()
|
}
|
||||||
.collect(HashMap::new,
|
|
||||||
(res, plant) -> res.put(plant.id(), plant),
|
@Override
|
||||||
(existing, replacement) -> { });
|
public Optional<Plant> getPlantById(long id) {
|
||||||
}
|
return Optional.empty();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,28 +7,7 @@ import java.io.IOException;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Optional;
|
import java.util.Optional;
|
||||||
|
|
||||||
/**
|
|
||||||
* A database of {@link Plant}s.
|
|
||||||
* The interface specifies the minimal required operations.
|
|
||||||
*/
|
|
||||||
public interface PlantDatabase {
|
public interface PlantDatabase {
|
||||||
/**
|
List<Plant> getPlantList(HardinessZone zone) throws IOException;
|
||||||
* Yields a list of all {@link Plant}s in the database with only data relevant to the specfied {@link HardinessZone}
|
Optional<Plant> getPlantById(long id) throws IOException;
|
||||||
*
|
|
||||||
* @param zone The zone for which data should be fetched
|
|
||||||
* @return A list of {@link Plant}s with data for the specified zone
|
|
||||||
* @throws IOException If the database cannot be accessed
|
|
||||||
* @throws HardinessZoneNotSetException If no {@link HardinessZone} was specified
|
|
||||||
*/
|
|
||||||
List<Plant> getPlantList(HardinessZone zone) throws IOException, HardinessZoneNotSetException;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Attempts to retrieve the {@link Plant} with the specified id.
|
|
||||||
*
|
|
||||||
* @param id The {@link Plant#id()} to look for
|
|
||||||
* @return {@link Optional} of the found {@link Plant}, {@link Optional#empty()} if no entry matched the criteria
|
|
||||||
* @throws IOException If the database cannot be accessed
|
|
||||||
* @throws HardinessZoneNotSetException If no {@link HardinessZone} was specified
|
|
||||||
*/
|
|
||||||
Optional<Plant> getPlantById(long id) throws IOException, HardinessZoneNotSetException;
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -12,7 +12,7 @@ public record GrowthPhase(
|
||||||
MonthDay startDate,
|
MonthDay startDate,
|
||||||
MonthDay endDate,
|
MonthDay endDate,
|
||||||
int group,
|
int group,
|
||||||
WateringCycle wateringCycle,
|
Object wateringCycle,
|
||||||
@JsonDeserialize(using = GrowthPhaseTypeDeserializer.class) GrowthPhaseType type,
|
@JsonDeserialize(using = GrowthPhaseTypeDeserializer.class) GrowthPhaseType type,
|
||||||
@JsonDeserialize(using = HardinessZoneDeserializer.class) HardinessZone zone,
|
@JsonDeserialize(using = HardinessZoneDeserializer.class) HardinessZone zone,
|
||||||
List<TaskTemplate> taskTemplates) {
|
List<TaskTemplate> taskTemplates) {
|
||||||
|
|
|
@ -1,4 +0,0 @@
|
||||||
package ch.zhaw.gartenverwaltung.types;
|
|
||||||
|
|
||||||
public record Pest(String name, String description, String measures) {
|
|
||||||
}
|
|
|
@ -1,46 +1,14 @@
|
||||||
package ch.zhaw.gartenverwaltung.types;
|
package ch.zhaw.gartenverwaltung.types;
|
||||||
|
|
||||||
import java.time.LocalDate;
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
import static java.time.temporal.ChronoUnit.DAYS;
|
|
||||||
|
|
||||||
public record Plant(
|
public record Plant(
|
||||||
long id,
|
long id,
|
||||||
String name,
|
String name,
|
||||||
String description,
|
String description,
|
||||||
String spacing,
|
int spacing,
|
||||||
int light,
|
int light,
|
||||||
String soil,
|
String soil,
|
||||||
List<Pest> pests,
|
List<Object> pests,
|
||||||
List<GrowthPhase> lifecycle) {
|
List<GrowthPhase> lifecycle) {
|
||||||
|
|
||||||
public void inZone(HardinessZone zone) {
|
|
||||||
lifecycle.removeIf(growthPhase -> !growthPhase.zone().equals(zone));
|
|
||||||
}
|
|
||||||
|
|
||||||
public List<GrowthPhase> lifecycleForGroup(int group) {
|
|
||||||
return lifecycle.stream()
|
|
||||||
.filter(growthPhase -> growthPhase.group() != group)
|
|
||||||
.toList();
|
|
||||||
}
|
|
||||||
|
|
||||||
public LocalDate sowDateFromHarvestDate(LocalDate harvestDate, int group) {
|
|
||||||
return harvestDate.minusDays(timeToHarvest(group));
|
|
||||||
}
|
|
||||||
|
|
||||||
public int timeToHarvest(int group) {
|
|
||||||
List<GrowthPhase> activeLifecycle = lifecycleForGroup(group);
|
|
||||||
GrowthPhase sow = activeLifecycle.stream()
|
|
||||||
.filter(growthPhase -> !growthPhase.type().equals(GrowthPhaseType.SOW))
|
|
||||||
.findFirst()
|
|
||||||
.orElseThrow();
|
|
||||||
GrowthPhase harvest = activeLifecycle.stream()
|
|
||||||
.filter(growthPhase -> !growthPhase.type().equals(GrowthPhaseType.HARVEST))
|
|
||||||
.findFirst()
|
|
||||||
.orElseThrow();
|
|
||||||
|
|
||||||
int currentYear = LocalDate.now().getYear();
|
|
||||||
return (int) DAYS.between(harvest.startDate().atYear(currentYear), sow.startDate().atYear(currentYear));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,8 +0,0 @@
|
||||||
package ch.zhaw.gartenverwaltung.types;
|
|
||||||
|
|
||||||
public record WateringCycle(
|
|
||||||
int litersPerSqM,
|
|
||||||
int interval,
|
|
||||||
String[] notes
|
|
||||||
) {
|
|
||||||
}
|
|
|
@ -81,174 +81,5 @@
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": 1,
|
|
||||||
"name": "Early Carrot",
|
|
||||||
"description": "Carrot, (Daucus carota), herbaceous, generally biennial plant of the Apiaceae family that produces an edible taproot. Among common varieties root shapes range from globular to long, with lower ends blunt to pointed. Besides the orange-coloured roots, white-, yellow-, and purple-fleshed varieties are known.",
|
|
||||||
"lifecycle": [
|
|
||||||
{
|
|
||||||
"startDate": "02-20",
|
|
||||||
"endDate": "03-10",
|
|
||||||
"zone": "ZONE_8A",
|
|
||||||
"type": "SOW",
|
|
||||||
"wateringCycle": {
|
|
||||||
"litersPerSqM": 15,
|
|
||||||
"interval": 3,
|
|
||||||
"notes": []
|
|
||||||
},
|
|
||||||
"taskTemplates": [
|
|
||||||
{
|
|
||||||
"name": "hilling",
|
|
||||||
"relativeStartDate": 0,
|
|
||||||
"relativeEndDate": 0,
|
|
||||||
"description": "Mound up the soil around the plant until just the top few leaves show above the soil. ",
|
|
||||||
"interval": null,
|
|
||||||
"isOptional": false
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"startDate": "03-10",
|
|
||||||
"endDate": "05-10",
|
|
||||||
"zone": "ZONE_8A",
|
|
||||||
"type": "PLANT",
|
|
||||||
"wateringCycle": {
|
|
||||||
"litersPerSqM": 25,
|
|
||||||
"interval": 3,
|
|
||||||
"notes": [
|
|
||||||
"Be careful not to pour water over the leaves, as this will lead to sunburn."
|
|
||||||
]
|
|
||||||
},
|
|
||||||
"taskTemplates": [
|
|
||||||
{
|
|
||||||
"name": "hilling",
|
|
||||||
"relativeStartDate": 0,
|
|
||||||
"relativeEndDate": null,
|
|
||||||
"description": "Mound up the soil around the plant until just the top few leaves show above the soil. ",
|
|
||||||
"interval": 15,
|
|
||||||
"isOptional": true
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"startDate": "05-10",
|
|
||||||
"endDate": "05-20",
|
|
||||||
"zone": "ZONE_8A",
|
|
||||||
"type": "HARVEST",
|
|
||||||
"wateringCycle": {
|
|
||||||
"litersPerSqM": 0,
|
|
||||||
"interval": null,
|
|
||||||
"notes": []
|
|
||||||
},
|
|
||||||
"taskTemplates": [
|
|
||||||
{
|
|
||||||
"name": "Harvesting",
|
|
||||||
"relativeStartDate": 0,
|
|
||||||
"relativeEndDate": 14,
|
|
||||||
"description": "When the leaves turn to a yellowish brown. Do not harvest earlier. The plant will show when it's ready.",
|
|
||||||
"interval": null,
|
|
||||||
"isOptional": false
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"soil": "sandy to loamy, loose soil, free of stones",
|
|
||||||
"spacing": "5,35,2.5",
|
|
||||||
"pests": [
|
|
||||||
{
|
|
||||||
"name": "Rot",
|
|
||||||
"description": "rot, any of several plant diseases, caused by any of hundreds of species of soil-borne bacteria, fungi, and funguslike organisms (Oomycota). Rot diseases are characterized by plant decomposition and putrefaction. The decay may be hard, dry, spongy, watery, mushy, or slimy and may affect any plant part.",
|
|
||||||
"measures": "less water"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"id": 2,
|
|
||||||
"name": "summertime onion",
|
|
||||||
"description": "Onion, (Allium cepa), herbaceous biennial plant in the amaryllis family (Amaryllidaceae) grown for its edible bulb. The onion is likely native to southwestern Asia but is now grown throughout the world, chiefly in the temperate zones. Onions are low in nutrients but are valued for their flavour and are used widely in cooking. They add flavour to such dishes as stews, roasts, soups, and salads and are also served as a cooked vegetable.",
|
|
||||||
"lifecycle": [
|
|
||||||
{
|
|
||||||
"startDate": "03-15",
|
|
||||||
"endDate": "04-10",
|
|
||||||
"type": "SOW",
|
|
||||||
"zone": "ZONE_8A",
|
|
||||||
"group": 0,
|
|
||||||
"wateringCycle": {
|
|
||||||
"litersPerSqM": 15,
|
|
||||||
"interval": 4,
|
|
||||||
"notes": [
|
|
||||||
|
|
||||||
]
|
|
||||||
},
|
|
||||||
"taskTemplates": [
|
|
||||||
{
|
|
||||||
"name": "hilling",
|
|
||||||
"relativeStartDate": 0,
|
|
||||||
"relativeEndDate": 0,
|
|
||||||
"description": "Mound up the soil around the plant until just the top few leaves show above the soil. ",
|
|
||||||
"interval": null,
|
|
||||||
"isOptional": false
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"startDate": "04-10",
|
|
||||||
"endDate": "07-10",
|
|
||||||
"type": "PLANT",
|
|
||||||
"zone": "ZONE_8A",
|
|
||||||
"group": 0,
|
|
||||||
"wateringCycle": {
|
|
||||||
"litersPerSqM": 25,
|
|
||||||
"interval": 3,
|
|
||||||
"notes": [
|
|
||||||
""
|
|
||||||
]
|
|
||||||
},
|
|
||||||
"taskTemplates": [
|
|
||||||
{
|
|
||||||
"name": "hilling",
|
|
||||||
"relativeStartDate": 0,
|
|
||||||
"relativeEndDate": null,
|
|
||||||
"description": "Mound up the soil around the plant until just the top few leaves show above the soil. ",
|
|
||||||
"interval": 15,
|
|
||||||
"isOptional": true
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"startDate": "07-10",
|
|
||||||
"endDate": "09-20",
|
|
||||||
"type": "HARVEST",
|
|
||||||
"zone": "ZONE_8A",
|
|
||||||
"group": 0,
|
|
||||||
"wateringCycle": {
|
|
||||||
"litersPerSqM": 0,
|
|
||||||
"interval": null,
|
|
||||||
"notes": [
|
|
||||||
|
|
||||||
]
|
|
||||||
},
|
|
||||||
"taskTemplates": [
|
|
||||||
{
|
|
||||||
"name": "Harvesting",
|
|
||||||
"relativeStartDate": 0,
|
|
||||||
"relativeEndDate": 14,
|
|
||||||
"description": "When ready for harvest, the leaves on your onion plants will start to flop over. This happens at the \"neck\" of the onion and it signals that the plant has stopped growing and is ready for storage. Onions should be harvested soon thereafter",
|
|
||||||
"interval": null,
|
|
||||||
"isOptional": false
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"soil": "sandy to loamy, loose soil, free of stones",
|
|
||||||
"spacing": "15,30,2",
|
|
||||||
"pests": [
|
|
||||||
{
|
|
||||||
"name": "Rot",
|
|
||||||
"description": "rot, any of several plant diseases, caused by any of hundreds of species of soil-borne bacteria, fungi, and funguslike organisms (Oomycota). Rot diseases are characterized by plant decomposition and putrefaction. The decay may be hard, dry, spongy, watery, mushy, or slimy and may affect any plant part.",
|
|
||||||
"measures": "less water"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
}
|
||||||
]
|
]
|
Loading…
Reference in New Issue