Skip to content
Snippets Groups Projects

Profile type

This section explains the profileType creation process in order to make different comparisons between the actual user's consumption and its profile type.

Profile type data

The profileType data are stored in the profile doctype and includes the following information :

{
  housingType: HousingType
  constructionYear: ConstructionYear
  area: string
  occupantsNumber: number
  outsideFacingWalls: OutsideFacingWalls
  floor: Floor
  heating: IndividualOrCollective
  coldWater: IndividualOrCollective
  hotWater: IndividualOrCollective
  individualInsulationWork: IndividualInsulationWork[]
  hasInstalledVentilation: ThreeChoicesAnswer
  hasReplacedHeater: ThreeChoicesAnswer
  hotWaterEquipment: HotWaterEquipment
  warmingFluid: WarmingType | null
  hotWaterFluid: FluidType | null
  cookingFluid: FluidType
  updateDate: DateTime
}

Each profileType value is an enum that you can find in the application. The values are set by default to a default profileType and needs to be completed by the user in order to become accurate.

Profile type form

User should answer a form to be able to set its profile type. Here is the flow of questions:

ProfileTypeForm

Profile types handling over time

A user can have many profileTypes, Ecolyo takes into account that one can move from his home, have more occupants, start renovating works and so on.

That's why a profileType is only active for a given time period, this is how its time logic works.

No filled profileType

A default profileType is loaded in redux to operate basic operations. When the user looks at its analysis or options, ecolyo suggest user to fill a more accurate profileType.

First profileType

The profileType will follow the form logic above (see ProfileType form section). updateDate field will be the current month and year. Since this profileType will be the only one in database, getProfileType() method will always return it, therefore it will be used for all past time period on analysis.

Following profileTypes

Each new profileType form will show a calendar picking section at the end of the form. By default, current month and year will be selected.

If current year is not yet completed (december not reached), month dropdown menu will only propose past and present months for this year.

When finishing, before the new profileType is saved. Checks if other profileTypes exist on the period between selected month/year and present day.

  • If profileTypes are found, they are deleted (newest profileType always prevails on older ones, so that user had forgotten to fill some information before, this new profiletype will be more accurate). Then save profileType on given month/year.
  • If no other profileTypes are found, save profileType on given month/year.

Profile type entity service

Here you can see the different methods provided by the profileType service. These methods allows to calculate the average consumption depending on the information gathered in user's profileType.

Method Description
getProfileType(date?: DateTime) : Promise-ProfileType, null Returns a profiletype closest to the given param date. When no param date, returns most recent profiletype
getAllProfileTypes(timePeriod?: TimePeriod) : Promise-ProfileType[], null Returns all profiletypes for given time period, if no param returns all profiletypes
updateProfileType(attributes: Partial-ProfileType): Promise-ProfileType, null Saves profiletype in CouchDB
deleteProfileTypes(profileTypes: ProfileType[]): Promise-boolean Delete array of profileTypes
parseProfileTypeEntityToProfileType(profileTypeEntity: ProfileType): ProfileType Retrieves ProfileType from the ProfileTypeEntity

Profile type service

Here you can see the different methods provided by the profileType service. These methods allows to calculate the average consumption depending on the information gathered in user's profileType.

Method Description
getMonthlyForecast(month: number) : Promise-MonthlyForecast Used to get the monthly forecast for a given month
getFluidForecast(fluidType: FluidType, month: number) : Promise-FluidForecast Used to get the monthly forecast for a given FluidType
getDetailsMonthlyForecast(fluidType: FluidType, month: number): Promise-DetailsMonthlyForecast Used to get the details of a fluidForecast. For each fluid return the related consumption
getMonthColdWaterConsumption(month: number): number Returns the cold water consumption in L for a given month
getMonthElectricSpecificConsumption(month: number): number Returns the electric specific consumption in kWh for a given month
getMonthCookingConsumption(month: number): number Returns the cooking consumption in kWh for a given month
getMonthEcs(month: number) : number Returns the hot water consumption in kWh for a given month
getMonthHeating(month: number): Promise-number Returns the heating consumption in kWh for a given month
calculateTotalConsumption(spreadConsumption: number, profileType: ProfileType, month: number) : number Returns the total ECS consumption with applyied corrections to the spreadConsumption depending on users facilities.
calculateSpreadNeeds(profileType: ProfileType, month: number): number Calculate the water spread consumption
calculateMonthWaterRawNeeds(profileType: ProfileType, month: number) : number Returns the raw water needs of a home
calculateWarmingEstimatedConsumption(): number Calculates the estimated consumption of a home
calculateWarmingCorrectedConsumption(estimatedConsumption: number): number Apply the different corrections to the estimated heating consumption
calculateWarmingMonthConsumption(correctedConsumption: number, month: number): Promise-number Apply dju ratio to the corrected heating consumption
fetchDJU(month: number): Promise-number Returns a Promise with the dju for the given month

Here is an example of a MonthlyForecast, which is the only function directly called by the analysis component :

{
  fluidForecast: [
    {
      detailsMonthlyForecast: {
        coldWaterConsumption: null,
        cookingConsumption: null,
        ecsConsumption: null,
        electricSpecificConsumption: 266,
        heatingConsumption: 4074,
      },
      fluidType: 0,
      load: 4340,
      value: 670.96,
    },
    {
      detailsMonthlyForecast: {
        coldWaterConsumption: 14911,
        cookingConsumption: null,
        ecsConsumption: null,
        electricSpecificConsumption: null,
        heatingConsumption: null,
      },
      fluidType: 1,
      load: 14911,
      value: 46.224,
    },
    {
      detailsMonthlyForecast: {
        coldWaterConsumption: null,
        cookingConsumption: 85,
        ecsConsumption: 290,
        electricSpecificConsumption: null,
        heatingConsumption: null,
      },
      fluidType: 2,
      load: 375,
      value: 29.737,
    },
  ],
  month: 1,
  totalValue: 746.921,
}

In these data, the load is in kWh for gas and electricity (fluidtype 0 and 2) and in L for water. Value and totalValue are given in euros. The different consumption are distributed to their corresponding fluidType according to the variables _warmingFluid, hotWaterFluid, cookingFluid _. For example, if the cookingFluid is gas, then its value will be stored in the fluidType 2 and nulled in the others.

Also, we consider that if the heating or the hotwater are collective, we don't calculate it because the user can never access his personal consumption, so it doesn't takes place in the calculation.

DJU

The method fetchDJU(month:number) tries to fetch the daily dju from the datagrandlyon API trough a remote doctype named org.ecolyo.dju. The data returned by this API are gathered by the lyon-bron-station The API returns dju by day for a given month, so we add them to return a monthly dju. If remote doctype fails or return a null value, we returns instead the default dju stored in heating.json

Useful tips while working with this API :

  • Do not pass a Content-Type while requesting, it will returns error
  • While requesting from a cozy instance, if you get errors, you will get errors from the cozy stack and not directly from the API, which leads to weird cases :
  • If there is an authentication error, the api will returns an error in html format to cozy, so you'll get from cozy an error 502 as following, because cozy expects a json response :
{
  "status": "502",
  "title": "Bad Gateway",
  "detail": "the content-type for the response is not authorized",
  "source": {}
}
  • In case of bad request (wrong parameters for example), the API will return an error in xml format so you'll get exactly the same error as above in your cozy instance.