EVCC is an extensible EV Charge Controller with PV integration implemented in Go.
- simple and clean user interface
- multiple chargers: Wallbe, Phoenix (includes ESL Walli), go-eCharger, NRGKick, SimpleEVSE, EVSEWifi, KEBA/BMW, openWB, Mobile Charger Connect, any other charger using scripting
- multiple meters: ModBus (Eastron SDM, MPM3PM, SBC ALE3 and many more), SMA Home Manager and SMA Energy Meter, KOSTAL Smart Energy Meter (KSEM, EMxx), any Sunspec-compatible inverter or home battery devices (Fronius, SMA, SolarEdge, KOSTAL, STECA, E3DC), Tesla PowerWall
- different vehicles to show battery status: Audi (eTron), BMW (i3), Tesla, Nissan (Leaf), any other vehicle using scripting
- plugins for integrating with hardware devices and home automation: Modbus (meters and grid inverters), MQTT and shell scripts
- status notifications using Telegram and PushOver
- logging using InfluxDB and Grafana
- soft ramp-up/ramp-down of charge current ensures contactor only switched at minimum current
- electric contactor protection
- REST API
EVCC is provided as binary executable file and docker image. Download the file for your platform and then execute like this:
evcc -h
Use the following systemd
unit description to configure EVCC as service (put into /etc/systemd/system/evcc.service
):
[Unit]
Description=evcc
After=syslog.target
[Service]
ExecStart=/usr/local/bin/evcc --log error
Restart=always
[Install]
WantedBy=multi-user.target
EVCC can also be run using Docker. Here's and example with given config file and UI on port 7070:
docker run -v $(pwd)/evcc.dist.yaml:/etc/evcc.yaml -p 7070:7070 andig/evcc -h
To build EVCC from source, Go 1.13 is required:
make
Note: EVCC comes without any guarantee. You are using this software entirely at your own risk. It is your responsibility to verify it is working as intended. EVCC requires a supported charger and a combination of grid, PV and charge meter. All components must be installed by a certified professional.
Multiple charge modes are supported:
- Off: disable the charger, even if car gets connected.
- Now (Sofortladen): charge immediately with maximum allowed current.
- Min + PV: charge immediately with minimum configured current. Additionally use PV if available.
- PV: use PV as available. May not charge the car if PV remains dark.
In general, due to the minimum value of 5% for signalling the EV duty cycle, the charger cannot limit the current to below 6A. If the available power calculation demands a limit less than 6A, handling depends on the charge mode. In PV mode, the charger will be disabled until available PV power supports charging with at least 6A. In Min + PV mode, charging will continue at minimum current of 6A and charge current will be raised as PV power becomes available again.
For both PV modes, EVCC needs to assess how much residual PV power is available at the grid connection point and how much power the charger actually uses. Various methods are implemented to obtain this information, with different degrees of accuracy.
-
PV meter: Configuring a PV meter is the simplest option. PV meter measures the PV generation. The charger is allowed to consume:
Charge Power = PV Meter Power - Residual Power
The pv meter is expected to deliver negative values for export and should not return positive values.
Residual Power is a configurable assumption how much power remaining facilities beside the charger use.
-
Grid meter: Configuring a grid meter is the preferred option. The grid meter is expected to be a two-way meter (import+export) and return the current amount of grid export as negative value measured in Watt (W). The charger is then allowed to consume:
Charge Power = Current Charge Power - Grid Meter Power - Residual Power
In this setup, residual power is used as margin to account for fluctuations in PV production that may be faster than EVCC's control loop.
-
Battery meter: battery meter is used if a home battery is installed and you want charging the EV take priority over charging the home battery. As the home battery would otherwise "grab" all available PV power, this meter measures the home battery charging power.
With grid meter the charger is then allowed to consume:
Charge Power = Current Charge Power - Grid Meter Power + Battery Meter Power - Residual Power
or without grid meter
Charge Power = PV Meter Power + Battery Meter Power - Residual Power
The battery meter is expected to deliver negative values when charging and positive values when discharging.
When using a grid meter for accurate control of PV utilization, EVCC needs to be able to determine the current charge power. There are two configurations for determining the current charge power:
-
Charge meter: A charge meter is often integrated into the charger but can also be installed separately. EVCC expects the charge meter to supply charge power in Watt (W) and preferably total energy in kWh. If total energy is supplied, it can be used to calculate the charged energy for the current charging cycle.
-
No charge meter: If no charge meter is installed, charge power is deducted from charge current as controlled by the charger. This method is less accurate than using a charge meter since the EV may chose to use less power than EVCC has allowed for consumption. If the charger supplies total energy for the charging cycle this value is preferred over the charge meter's value (if present).
The EVCC consists of four basic elements: Charger, Meter and Vehicle individually configured and attached to Loadpoints.
Charger is responsible for handling EV state and adjusting charge current. Available charger implementations are:
wallbe
: Wallbe Eco chargers (see Preparation). For older Wallbe boxes (pre 2019) with Phoenix EV-CC-AC1-M3-CBC-RCM-ETH controllers make sure to setlegacy: true
to enable correct current configuration.phoenix-emcp
: chargers with Phoenix EM-CP-PP-ETH controllers like the ESL Walli (Ethernet connection).phoenix-evcc
: chargers with Phoenix EV-CC-AC1-M controllers (ModBus connection)simpleevse
: chargers with SimpleEVSE controllers connected via ModBus (e.g. OpenWB)evsewifi
: chargers with SimpleEVSE controllers using SimpleEVSE-Wifinrgkick
: NRGKick chargers with Connect modulego-e
: go-eCharger chargerskeba
: KEBA KeContact P20/P30 and BMW chargers (see Preparation)mcc
: Mobile Charger Connect devices (Audi, Bentley, Porsche)default
: default charger implementation using configurable plugins for integrating any type of charger
Wallbe chargers are supported out of the box. The Wallbe must be connected using Ethernet. If not configured, the default address 192.168.0.8:502
is used.
To allow controlling charge start/stop, the Wallbe physical configuration must be modified. This requires opening the Wallbe. Once opened, DIP 10 must be set to ON:
More information on interacting with Wallbe chargers can be found at GoingElectric. Use with care.
NOTE: The Wallbe products come in two flavors. Older models (2017 known to be "old", 2019 known to be "new") use the Phoenix EV-CC-AC1-M3-CBC-RCM controller. For such models make sure to set legacy: true
. You can find you which one you have using MBMD:
mbmd read -a 192.168.0.8:502 -d 255 -t holding -e int 300 1
Compare the value to what you see as Actual Charge Current Setting in the Wallbe web UI. If the numbers match, it's a Phoenix controller, if the reading is factor 10x the UI value then it's a Wallbe controller.
NOTE: Opening the wall box must only be done by certified professionals. The box must be disconnected from mains before opening.
KEBA chargers require UDP function to be enabled with DIP switch 1.3 = ON
, see KEBA installation manual.
If using Docker, make sure that the Docker container can receive UDP messages on port 7090 used by KEBA by using host networking in Docker:
docker run --network=host -p 7070:7070 andig/evcc ...
Meters provide data about power and energy consumption. Available meter implementations are:
-
modbus
: ModBus meters as supported by MBMD. Configuration is similar to the ModBus plugin wherepower
andenergy
specify the MBMD measurement value to use:- name: pv type: modbus model: sdm uri: rs485.fritz.box:23 id: 2 power: Power # reading as understood by MBMD, leave empty for power default value energy: Export # optional reading for total energy values, specify for charge meter
-
sma
: SMA Home Manager and SMA Energy Meter. Power reading is configured out of the box but can be customized if necessary. To obtain energy readings define the desired Obis code (Import Energy: "1:1.8.0", Export Energy: "1:2.8.0"):- name: sma-home-manager type: sdm uri: 192.168.1.4 power: # leave empty for combined import/export power choose obis energy: # leave empty to disable or choose obis 1:1.8.0/1:2.8.0
-
tesla
: Tesla PowerWall meter. Usevalue
to choose meter (grid meter:site
, pv:solar
, battery:battery
)- name: powerwall type: tesla uri: http://192.168.1.4/api/meters/aggregates meter: site # grid meter: `site`, pv: `solar`, battery: `battery`
Note: this could also be implemented using a
default
meter with thehttp
plugin. -
default
: default meter implementation where meter readings-power
andenergy
are configured using plugin- name: vzlogger type: default power: type: http # or any other plugin ... energy: type: http # or any other plugin ...
Vehicle represents a specific EV vehicle and its battery. If vehicle is configured and assigned to the charger, charge status and remaining charge duration become available in the user interface.
Available vehicle implementations are:
audi
: Audi (eTron)bmw
: BMW (i3)nissan
: Nissan (Leaf)tesla
: Tesla (any model)renault
: Renault (Zoe, Kangoo ZE)default
: default vehicle implementation using configurable plugins for integrating any type of vehicle
Plugins are used to integrate physical devices and external data sources with EVCC. Plugins support both read and write access. When using plugins for write access, the actual data is provided as variable in form of ${var[:format]}
. If format
is omitted, data is formatted according to the default Go %v
format. The variable is replaced with the actual data before the plugin is executed.
The calc
plugin allows calculating the sum of other plugins:
type: calc
add:
- type: ...
...
- type: ...
...
The calc
plugin is useful e.g. to combine power values if import and export power are separate like with S0 meters. Use scale
on one of the elements to implement a subtraction.
The modbus
plugins is able to read data from any Modbus meter or SunSpec-compatible solar inverter. Many meters are already pre-configured (see MBMD Supported Devices).
The meter configuration consists of the actual physical connection and the value to be read.
If the device is physically connected using an RS485 adapter, device
and serial configuration baudrate
, comset
must be specified:
type: modbus
device: /dev/ttyUSB0
baudrate: 9600
comset: "8N1"
If the device is a grid inverter or a Modbus meter connected via TCP, uri
must be specified:
type: modbus
uri: 192.168.0.11:502
id: 1 # modbus slave id
If the device is a Modbus RTU device connected using an RS485/Ethernet adapter, set rtu: true
. The serial configuration must be done directly on the adapter. Example:
type: modbus
uri: 192.168.0.10:502
id: 3 # modbus slave id
rtu: true
The meter device type meter
and the device's slave id id
are always required:
type: ...
uri/device/id: ...
model: sdm
value: Power
scale: -1 # floating point factor applied to result, e.g. for kW to W conversion
Supported meter models are the same as supported by MBMD:
- RTU:
ABB
ABB A/B-Series metersMPM
Bernecker Engineering MPM3PM metersDZG
DZG Metering GmbH DVH4013 metersINEPRO
Inepro Metering Pro 380JANITZA
Janitza B-Series metersSBC
Saia Burgess Controls ALD1 and ALE3 metersSDM
Eastron SDM630SDM220
Eastron SDM220SDM230
Eastron SDM230SDM72
Eastron SDM72
- TCP: Sunspec-compatible grid inverters (SMA, SolarEdge, KOSTAL, Fronius, Steca etc)
Use value
to define the value to read from the device. All values that are supported by MBMD are pre-configured.
If the Modbus device is not supported by MBMD, the Modbus register can also be manually configured:
type: ...
uri/device/id: ...
register:
address: 40070
length: 2 # read length in words
type: holding # holding or input
decode: int32 # int16|32|64, uint16|32|64, float32|64 and u|int32s
scale: -1 # floating point factor applied to result, e.g. for kW to W conversion
The int32s/uint32s
decodings apply swapped word order and are useful e.g. with E3/DC devices.
The mqtt
plugin allows to read values from MQTT topics. This is particularly useful for meters, e.g. when meter data is already available on MQTT. See MBMD for an example how to get Modbus meter data into MQTT.
Sample configuration:
type: mqtt
topic: mbmd/sdm1-1/Power
timeout: 30s # don't accept values older than timeout
scale: 0.001 # floating point factor applied to result, e.g. for kW to W conversion
Sample write configuration:
type: mqtt
topic: mbmd/charger/maxcurrent
payload: ${var:%d}
For write access, the data is provided using the payload
attribute. If payload
is missing, the value will be written in default format.
The script
plugin executes external scripts to read or update data. This plugin is useful to implement any type of external functionality.
Sample read configuration:
type: script
cmd: /bin/bash -c "cat /dev/urandom"
timeout: 5s
Sample write configuration:
type: script
cmd: /home/user/my-script.sh ${enable:%b} # format boolean enable as 0/1
timeout: 5s
The http
plugin executes HTTP requests to read or update data. Includes the ability to read and parse JSON using jq-like queries.
Sample read configuration:
type: http
uri: https://volkszaehler/api/data/<uuid>.json?from=now
method: GET # default HTTP method
headers:
- content-type: application/json
insecure: false # set to true to trust self-signed certificates
jq: .data.tuples[0][1] # parse response json
scale: 0.001 # floating point factor applied to result, e.g. for kW to W conversion
Sample write configuration:
...
body: %v # only applicable for PUT or POST requests
The combined
status plugin is used to convert a mixed boolean status of plugged/charging into an EVCC-compatible charger status of A..F. It is typically used together with OpenWB MQTT integration.
Sample configuration (read only):
type: combined
plugged:
type: mqtt
topic: openWB/lp/1/boolPlugStat
charging:
type: mqtt
topic: openWB/lp/1/boolChargeStat
EVCC has the following internal API. The full documentation is available in GoDoc format in https://pkg.go.dev/github.com/andig/evcc/api.
Status()
: get charge controller status (A...F
)Enabled()
: get charger availabilityEnable(bool)
: set charger availabilityMaxCurrent(int)
: set maximum allowed charge current in A
Optionally, charger can also provide:
CurrentPower()
: power in W (used if charge meter is not present)
CurrentPower()
: power in WTotalEnergy()
: energy in kWh (optional)
Title()
: vehicle name for display in the configuration UICapacity()
: battery capacity in kWhChargeState()
: state of charge in %
Optionally, vehicles can also provide:
CurrentPower()
: charge power in W (used if charge meter not present)ChargedEnergy()
: charged energy in kWhChargeDuration()
: charge duration
EVCC is heavily inspired by OpenWB. However, I found OpenWB's architecture slightly intimidating with everything basically global state and heavily relying on shell scripting. On the other side, especially the scripting aspect is one that contributes to OpenWB's flexibility.
Hence, for a simplified and stricter implementation of an EV charge controller, the design goals for EVCC were:
- typed language with ability for systematic testing - achieved by using Go
- structured configuration - supports YAML-based config file
- avoidance of feature bloat, simple and clean UI - utilizes Bootstrap
- containerized operation beyond Raspberry Pi - provide multi-arch Docker Image
- support for multiple load points - tbd