CDR Forwarding

⚠️

CDR Forwarding is only available when external clearing is chosen as a payment option. When Plugsurfing is the acting payment provider, the session is used as a record of the charging session.

Introduction

A CDR (Charging Detail Record) contains information about a completed charging session.

Overall, they contain the following information:

  • Who: user or charging key information
  • What: details on the charging session, including cost details
  • Where: location details

CDRs are sent to you using webhooks.

CDRs on Drive API

  • Requests will be sent as HTTP POST
  • The client should return a 200 response when the request is accepted
  • The client should return a 400 response when the request is not valid
  • The client should return a 403 response if authorization failed (see Configuration section below)
  • The client should return a 500 response in case it fails
  • We expect a response within 5 seconds. Any processing that can lead to response times higher than 5 seconds should be done asynchronously
  • New fields might be added to the schema without prior notice. Ensure that unknown fields are ignored and do not result in errors.

Retry logic

If the client doesn't respond with 200 or doesn't respond in time, the request will be retried: the CDR will be resent every night for 7 days (both on stage and prod environments).

Idempotency Expectation

For reliability, each webhook request includes a unique requestId that remains the same across all retry attempts. Therefore, your system should handle webhook events idempotently. This means:

  • Ensure your system processes each event only once even if you receive the same requestId multiple times due to retries.
  • Return a200 response as soon as you have successfully processed the event or stored it for later processing.

Example Scenario:

  • We send a requet to your endpoint, your system experiences a delay and does not return a response in time so we mark it as failed, but your system eventually manages to process it.
  • We will resend the same event again later, including the same requestId. At that point, we expect your system to recognize that this event has already been processed, so you should return a 200 response.

Handling retries idempotently helps avoid duplicate processing, ensuring consistent behavior across webhook interactions.

Configuration

Here is what we need to configure the CDR webhook:

  • URL to which we will send the CDRs
  • Authorization headers that will be passed in the request to the mentioned URL

Please contact your account manager with this data, so that we can set it up.

CDR Data

fieldtypemandatorynotes
requestIdStringyesrequestId will not change upon retry
timestampString ISO date formatyesexample: 2022-12-12T12:55:21Z (always UTC)
payerObject (Payer)yessee Payer section below
typeString / Enumyespossible value: SESSION
itemObject (Item)yessee Item section below

Payer Model

There are two possible Payer models depending on the integration:

User Model

fieldtypemandatorynotes
typeString / EnumyesUSER
idStringyesUser id (provided during user creation)
chargingKeyObjectyes
chargingKey.visualNumberStringno
chargingKey.uidStringyes
chargingKey.contractIdStringyes
chargingKey.identificationTypeStringyesPossible values: rfid, virtual or plug_and_charge

See User model example for a sample JSON payload.

Charging Key Model

fieldtypemandatorynotes
typeString / EnumyesCHARGING_KEY
uidStringyeskey identifier of the charging key paying
visualNumberStringnoabsent for virtual keys
contractIdStringyes
identificationTypeStringyesPossible values: rfid, virtual or plug_and_charge

See Charging key model example for a sample JSON payload.

Item

Session

fieldtypemandatorynotes
sessionIdStringyes
externalSessionIdStringnosession id received from the CPO
startTimeString ISO date formatyessession startTime, always UTC
stopTimeString ISO date formatyessession stopTime, always UTC
energyConsumedInWhLongyestotal energy consumed during the session, in Wh
totalParkingTimeInSecondsLongnototal parking time in seconds (if available): duration during which the car was not charging. To calculate the total charging time:stopTime - startTime - totalParkingTimeInSeconds
empCostObject (Cost)yesend-customer cost (“Cost” object described below)
cpoClaimedCostObject (Cost)yescost received from the CPO
locationObject (Location)yes
homeChargingObject (HomeCharging)noOnly present in the case of homeCharging
Location
fieldtypemandatorynotes
idStringyeslocationId
evseIdStringyese.g. IT_TSL_EMIL*0001
cpoIdStringyesoperator's external id (e.g. "DE*ABC")
addressStringyes
cityStringyes
postalCodeStringno
countryStringyesISO code (3 letters) of the country
powerTypeStringyesPossible values: AC,DC,AC_1_PHASE,AC_3_PHASE
privateBooleanyeswhether the location is private or not
coordinatesObject(GeoCoordinates)yesthe location's geo coordinates
GeoCoordinates
fieldtypemandatory
latitudeDoubleyes
longitudeDoubleyes
Cost
fieldtypemandatorynotes
totalCostMinorUnitsInclVatLongyestotal cost including VAT in minor units
totalCostMinorUnitsExclVatLongyestotal cost excluding VAT in minor units
currencyStringyesCurrency ISO code (3 letters)
costSegmentsArray of CostSegmentyes
CostSegment
fieldtypemandatorynotes
typeString / EnumyesPossible values: TIME, ENERGY, FLAT, CAPPED, PARKING_TIME
quantityDoubleyesamount of unit objects
vatRateDoubleyes
pricePerUnitMinorUnitsInclVatLongyesprice per unit including VAT in minor units
segmentCostMinorUnitsExclVatLongyessegment cost excluding VAT in minor units
segmentCostMinorUnitsInclVatLongyessegment cost including VAT in minor units
unitObject Unityesmin or h for TIME and PARKING_TIME, Wh or kWh for ENERGY, session for FLAT and CAPPED
fromString ISO date formatno*2022-12-12T12:55:21Z always UTC * - mandatory for TIME, PARKING_TIME and ENERGY types
toString ISO date formatno*2022-12-12T12:55:21Z always UTC * - mandatory for TIME, PARKING_TIME and ENERGY types
Minor units

Minor units are the smallest subdivisions of each currency.
Here are the specifications for handling costs in various currencies:

Currency CodeCurrency NameMinor UnitExample Cost
EUREuroCents100 cents → €1.00
SEKSwedish KronaÖre100 öre → 1 SEK
DKKDanish KroneØre100 øre → 1 DKK
NOKNorwegian KroneØre100 öre → 1 NOK
GBPBritish PoundPence100 pence → £1.00

For more information, check minor units explanation.

HomeCharging
fieldtypemandatorynotes
exemptVatBooleanyesif true, this Home Charging session is exempt from VAT.
Unit

Describes what quantity in PriceElement refers to. For ENERGY, FLAT and CAPPED the Unit doesn’t change, but for TIME and PARKING_TIME it can differ.

fieldtypemandatorynotes
typeStringyessession, kWh or min
amountnumber(integer)yesAlways 1 for session and kWh. Might be different (1 / 15 / 60 or other) for minutes.

segmentPriceMinorUnitsInclVat contains the total price for this segment.
It’s calculated as quantity * pricePerUnitMinorUnitsInclVat.

In the example below, the user charged over one hour. With a step size of one hour (60 minutes), this results in a quantity of two ‘steps’. See Prices for more details on how step size works.

Example PriceElement

{
  "type": "TIME",
  "quantity": 2,
  "pricePerUnitMinorUnitsInclVat": 12600,
  "segmentPriceMinorUnitsInclVat": 25200,
  "unit": {
    "type": "min",
    "amount": 60
  },
  "from": "2023-06-13T12:55:21Z",
  "to": "2023-06-13T14:22:33Z"
}

In priceElements array the same type might show up multiple times (example: the price is 1€/min from 00:00 to 12:00 and 0.5€/min from 12:00 to 00:00, session lasted from 11:30 till 13:00. CDR will contain two TIME price elements).

Examples

User model example

{
  "requestId": "ej4KDd2kKdj",
  "timestamp": "2024-09-25T13:44:27Z",
  "payer": {
    "id": "FJEDK34KSJ9WD",
    "chargingKey": {
      "uid": "NR9348593",
      "contractId": "DE-8PS-C1Z3R5KZI-8",
      "visualNumber": "2340-1254-6543-5655",
      "identificationType": "rfid",
      "type": "CHARGING_KEY"
    },
    "type": "USER"
  },
  "type": "SESSION",
  "item": {
    "sessionId": "BpK64y1z1QA",
    "externalSessionId": "89111162",
    "startTime": "2024-09-25T13:17:57Z",
    "stopTime": "2024-09-25T13:44:18Z",
    "empCost": {
      "totalCostMinorUnitsInclVat": 26594,
      "totalCostMinorUnitsExclVat": 21275,
      "currency": "NOK",
      "costSegments": [
        {
          "type": "ENERGY",
          "quantity": 44.416,
          "pricePerUnitMinorUnitsInclVat": 599,
          "segmentCostMinorUnitsExclVat": 21275,
          "segmentCostMinorUnitsInclVat": 26594,
          "unit": {
            "type": "kWh",
            "amount": 1
          },
          "vatRate": 25.0,
          "from": "2024-09-25T13:17:57Z",
          "to": "2024-09-25T13:44:18Z"
        }
      ]
    },
    "cpoClaimedCost": {
      "totalCostMinorUnitsInclVat": 25275,
      "totalCostMinorUnitsExclVat": 20220,
      "currency": "NOK",
      "costSegments": [
        {
          "type": "FLAT",
          "quantity": 1.0,
          "pricePerUnitMinorUnitsInclVat": 25275,
          "segmentCostMinorUnitsExclVat": 20220,
          "segmentCostMinorUnitsInclVat": 25275,
          "unit": {
            "type": "session",
            "amount": 1
          },
          "vatRate": 25.0
        }
      ]
    },
    "energyConsumedInWh": 44416,
    "totalParkingTimeInSeconds": 0,
    "location": {
      "id": "1ih2FfSRddmxioJEmsyDcv9a2h+e83m568C3LyoNkIs=",
      "evseId": "NO*CHA*E2496*A",
      "cpoId": "NO*REC",
      "address": "Hardangervegen 697",
      "city": "Haukeland",
      "postalCode": "5268",
      "country": "NOR",
      "powerType": "DC",
      "private": false
    }
  }
}

Charging key model example

{
  "requestId": "fjei9pw0c",
  "timestamp": "2024-09-25T15:00:51Z",
  "payer": {
    "uid": "93042D4B7AD96280",
    "contractId": "GB-ALS-EF90LT13L-O",
    "visualNumber": "RFID34957483933",
    "identificationType": "rfid",
    "type": "CHARGING_KEY"
  },
  "type": "SESSION",
  "item": {
    "sessionId": "mezoOeWGdmpa",
    "externalSessionId": "GBKR40G9F92JDKS",
    "startTime": "2024-09-25T13:54:42Z",
    "stopTime": "2024-09-25T15:00:46Z",
    "empCost": {
      "totalCostMinorUnitsInclVat": 2380,
      "totalCostMinorUnitsExclVat": 1983,
      "currency": "GBP",
      "costSegments": [
        {
          "type": "ENERGY",
          "quantity": 30.131,
          "pricePerUnitMinorUnitsInclVat": 79,
          "segmentCostMinorUnitsExclVat": 1983,
          "segmentCostMinorUnitsInclVat": 2380,
          "unit": {
            "type": "kWh",
            "amount": 1
          },
          "vatRate": 20.0,
          "from": "2024-09-25T13:54:42Z",
          "to": "2024-09-25T15:00:41Z"
        }
      ]
    },
    "cpoClaimedCost": {
      "totalCostMinorUnitsInclVat": 2380,
      "totalCostMinorUnitsExclVat": 1984,
      "currency": "GBP",
      "costSegments": [
        {
          "type": "FLAT",
          "quantity": 1.0,
          "pricePerUnitMinorUnitsInclVat": 2380,
          "segmentCostMinorUnitsExclVat": 1984,
          "segmentCostMinorUnitsInclVat": 2380,
          "unit": {
            "type": "session",
            "amount": 1
          },
          "vatRate": 20.0
        }
      ]
    },
    "energyConsumedInWh": 30131,
    "totalParkingTimeInSeconds": 4,
    "location": {
      "id": "pdM9JM8qNtM3MX761qOGUlhljKJ/n6l1lfXMU2w0AivH5+3bUvPdMZyqaUV98hyfgaLCza2kwLfKutDcynXGpQ==",
      "evseId": "GB*OSP*EOSP20191*2",
      "cpoId": "GB*OSP",
      "address": "Weedon Bec",
      "city": "Northampton",
      "postalCode": "NN7 4QD",
      "country": "GBR",
      "powerType": "DC",
      "private": false
    }
  }
}

Security

If you want to be sure that CDRs you receive are emitted by Plugsurfing, please check the Verify HMAC Signatures guide.