Akeneo Measure Bundle : manage measure units in families and conversions from a unit to another
Allows to :
- Convert a value from a unit to another
- Add more unit to a family (group of measure units)
- Create new families
Originally written to be used in Akeneo PIM.
This extraction aims to make it usable outside of the Akeneo PIM project.
Please consider that this process is still in a very experimental stage.
We don't have all the tests and automation which could ensure that this component will work smoothly outside of the Akeneo PIM full stack.
Documentation is available on docs.akeneo.com
This bundle to convert a value from a unit to another. Some families and units are already defined but it's possible to add others families and units to convert anything.
Converter converts a value from a unit to a standard unit (sort of reference unit) then from standard unit to asked unit. This allows to define just one list of operations to convert each unit.
Operations are defined to convert from the unit to the standard unit. Converter convert to standard unit with default operations order but reverse order and do the opposite operation (addition <=> substraction and multiplication <=> division). We used strings to define operations. 'add', 'sub', mul', 'div' are allowed.
You can add more operations extending bundle.
In MeasureBundle :
- Convert/ contains the converter service.
- DependencyInjection/
- Configuration : define how to check configuration measure files
- AkeneoMeasureExtension : define the recovery of the configuration (config files browse and merge configuration).
- Exception/ contains the exception classes used in converter.
- Family/ contains a list of family measure interfaces. Each define a family and must contains a constant named FAMILY.
- Resources/config/
- measure.yml : define measure configuration with a list of families. Each family define a standard unit measure and a list of unit measures. Each measure of this list is defined by a constant then by a conversion rule (list of ordered operations from unit to standard) and a symbol to display.
- services.yml : define measure converter service
parameters:
akeneo_measure.measures_config: ~
services:
akeneo_measure.measure_converter:
class: Akeneo\Bundle\MeasureBundle\Convert\MeasureConverter
arguments: [%akeneo_measure.measures_config%]
Configuration file can be seen https://github.com/akeneo/MeasureBundle/blob/master/Resources/config/measure.yml
To run tests :
$ php composer.phar update --dev
$ phpunit --coverage-html=cov/
A service is defined to use converter. You must call it and define the family to use before convert a value. In the example below, we convert a value in kilometers to miles.
$converter = $this->container->get('akeneo_measure.measure_converter');
$converter->setFamily(LengthFamilyInterface::FAMILY);
$result = $converter->convert(LengthFamilyInterface::KILOMETER, LengthFamilyInterface::MILE, 1);
$this->container->getParameter('akeneo_measure.measures_config');
To define a new unit in an existing family, it's just necessary to define it and their units in a new config file named measure.yml in your own bundle. For example, in our demo bundle, we add the below code :
measures_config:
Length:
standard: METER
units:
DONG:
convert: [{'mul': 7},{'div': 300}]
symbol: dong
Here, we just had "Dong" unit with his conversion rules from it to standard unit. To have equivalent to 1 dong in meters, you must multiply by 7 and divide by 300. A symbol is required too to define unit format to display. Optionally but recommended, a new class extending family class can be created. It allows to use converter with constants instead of strings. Contants represent config values. Here we created "MyLengthMeasure" new class extending LengthMeasure to add "Dong" unit constant.
use Akeneo\Bundle\MeasureBundle\Family\LengthFamilyInterface;
/**
* Override LengthFamily interface to add Dong measure constant
*/
class MyLengthFamilyInterface extends LengthFamilyInterface
{
/**
* @staticvar string
*/
const DONG = 'DONG';
}
Then, you can call a conversion to your new unit like this :
$converter = $this->container->get('akeneo_measure.measure_converter');
$converter->setFamily(LengthFamilyInterface::FAMILY);
$result = $converter->convert(LengthFamilyInterface::KILOMETER, MyLengthFamilyInterface::DONG, 1);
To create a new family, it's like to add a unit to an existing family. It's necessary to add configuration in measure.yml file of your bundle and optionally a class defining constants to be used instead of strings.
measures_config:
Capacitance:
standard: FARAD
units:
FARAD:
convert: [{'mul': 1}]
symbol: F
KILOFARAD:
convert: [{'mul': 1000}]
symbol: kF
MEGAFARAD:
convert: [{'mul': 1000000}]
symbol: MF
/**
* Capacitance measures constants
*/
class CapacitanceFamilyInterface
{
/**
* Family measure name
* @staticvar string
*/
const FAMILY = 'Capacitance';
/**
* @staticvar string
*/
const FARAD = 'FARAD';
/**
* @staticvar string
*/
const KILOFARAD = 'KILOFARAD';
/**
* @staticvar string
*/
const MEGAFARAD = 'MEGAFARAD';
}
Exceptions are thrown if we encountered problems during conversion.
- UnknownFamilyMeasureException if you try to use an unexistent or undefined family.
- UnknownMeasureException if you try to convert an unexistent or undefined unit for the family used.
- UnknownOperatorException if you try to use an unexistent operation (authorized add, sub, mul and div)
Divisions by zero don't throw exceptions but are ignored.
This bundle is extensible and we can imaginate config recovering from database or services or use converter for currencies for example.
This bundle is developed in our main repository.
If you encounter any bug, please check if it's already known before to create a new issue.
If you want to contribute (and we will be pleased if you do!), you'll find more information on this page.
About versioning, the version 0.6.x is developed in akeneo/pim-community-dev 1.6.x.
For older versions,
- akeneo/pim-community-dev 1.1.x uses akeneo/measure-bundle 0.1.x
- akeneo/pim-community-dev 1.2.x uses akeneo/measure-bundle 0.2.x
- akeneo/pim-community-dev 1.3.x uses akeneo/measure-bundle 0.3.x
- akeneo/pim-community-dev 1.4.x uses akeneo/measure-bundle 0.4.x
- akeneo/pim-community-dev 1.5.x uses akeneo/measure-bundle 0.5.x
- akeneo/pim-community-dev 1.6.x re-integrates this bundle in our main repository
Licence can be found Here.