Product Management ****************** A product in butler system is the final entity uniquely identified by schema of it's category or an overridden schema at the product level. Products can be created without creating a parent category or catalog, however it is advised otherwise. Item Object =========== .. gm:data:: item .. gm:data:: schema *optional dict* A :ref:`Schema Object` that defines the schema that this product will follow. If not given, it will inherit the schema of it’s category. Schema **must** be given either at category or product level. .. gm:data:: clientCategoryId *optional string (255)* A string id of the category to which this product belongs. .. gm:data:: attributes *required dict* This is an object containing some product metadata that are required, like weight and dimensions. It's required structure is: .. code-block:: json { "attributes": { "weight": { "uom": { "unit": "KG" }, "value": 1 }, "dimension": { "uom": { "unit": "CM" }, "length": 24, "width": 1, "height": 7 } } } :abbr:`UOM (Unit of Measure)`\ 's unit can take the following values: * MM - Millimeter * CM - Centimeter * M - Meter * INCH - Inches * GM - Gram * KG - Kilogram * LBS - Pounds * POUND - Pounds * OUNCE - Ounces .. gm:data:: productAttributes *required dict* This is an object containing key-values pairs that describe the product and the system's behavior for the product. The attribute's data type and (optionally) display string in different locales should be defined at the catalog level. The only **required** attribute inside this is ``product_sku``. Also, all groups of attributes defined as "unique" in the product's schema (or inheited schema) **must** be unique. Please check :ref:`Product Feature Attributes` for a list of attributes that affect product handling and other features in the system. .. gm:data:: barcodeList *required list* This is a list of barcode objects that **uniquely** identify this product. Each barcode in the barcode objects in this list must be globally unique for the whole system. It's structure should be like this: .. code-block:: json { "barcodeList": [ { "barcode": "Item1Barcode" } ] } .. gm:data:: description *optional string(255)* An optional string describing the product. .. gm:data:: imageUrl *optional string (255)* Should be an accessible link to an image representing the product. .. gm:data:: itemConfig *optional dict* An :ref:`Item Config Object` that defines the storage levels' configuration of the product. Item Object Sample JSON ----------------------- .. versionchanged:: 3.3-s60 itemConfig's attributes, quantity, and barcodeList changed .. literalinclude:: item_object.json :language: JSON Item Config Object ================== This object can be used to describe the storage levels for a product. For example, if a product comes in a case of 10 eaches, then this metadata can be stored in the product configuration so that the box dimensions, weight, and other attributes don't have to be passed in each request. Item config is defined as a recursive object. Each level defines the type, barcode, quantity, and other attributes for that level. .. gm:data:: itemConfig .. gm:data:: levelType *autogenerated integer* This is a unique id for this level .. gm:data:: label *required string* Name of this storage level, like ``case``, ``pallet``, ``sub-pack``, etc. .. gm:data:: type *required string* This is a string indicating the type of this level. Possible values: * ``marked``: Container with a barcode * ``unmarked``: Container without a barcode * ``virtual``: Not a container (i.e, an each) .. gm:data:: quantity .. versionchanged:: 3.3-s60 Renamed from ``qty`` to ``quantity`` *required integer* This denotes the quantity of objects in this level that is present in it's parent level. For root levels, this value must be 1. For example, for the configuration: "10 eaches in case", the quantity at case level would be 1, and at each level would be 10. .. gm:data:: barcodeList .. versionchanged:: 3.3-s60 Renamed from ``barcode`` to ``barcodeList`` *optional list* This is a list of barcode objects that uniquely identify this level. Each barcode in the barcode objects in this list must be globally unique for the whole system. It’s structure should be like this: .. code-block:: json { "barcodeList": [ { "barcode": "case1_barcode" } ] } .. gm:data:: level *optional list* This is an optional list of child level objects to this level. .. gm:data:: attributes .. versionchanged:: 3.3-s60 Structure has been changed to match :gm:data:`item.attributes`\ 's structure *required dict* This is an object containing some metadata that is required, like weight and dimensions. It's required structure is: .. code-block:: json { "attributes": { "weight": { "uom": { "unit": "KG" }, "value": 1 }, "dimension": { "uom": { "unit": "CM" }, "length": 24, "width": 1, "height": 7 } } } .. note:: In case of eaches, these values are ignored, and those specified in :gm:data:`item.attributes` are used. .. gm:data:: weight .. versionchanged:: 3.3-s60 Type has been changed from integer to dictionary. See above for example. .. note:: **For versions 3.3-s58 and before** *required integer* Weight of the level in the unit specified by :gm:data:`itemConfig.attributes.WeightUOM`. In case of containers (non-each levels), this should be the weight of the container box itself, excluding the weight of items inside it. In case of eaches, this value is ignored and the weight value is taken from the :ref:`Item Object`. .. gm:data:: dimension .. versionadded:: 3.3-s60 See above for example. .. gm:data:: length, width, height .. deprecated:: 3.3-s60 Use :gm:data:`itemConfig.attributes.dimension` instead *required integers* Dimensions of the level in the unit specified by :gm:data:`itemConfig.attributes.DimensionUOM`. In case of eaches, this value is ignored and the dimensions are taken from the :ref:`Item Object`. .. gm:data:: DimensionUOM .. deprecated:: 3.3-s60 Use :gm:data:`itemConfig.attributes.dimension` instead *required string* :abbr:`UOM (Unit of Measure)` for the dimensions. * MM - Millimeter * CM - Centimeter * M - Meter * INCH - Inches .. gm:data:: WeightUOM .. deprecated:: 3.3-s60 Use :gm:data:`itemConfig.attributes.weight` instead *required string* :abbr:`UOM (Unit of Measure)` for the weight. * GM - Gram * KG - Kilogram * LBS - Pounds * POUND - Pounds * OUNCE - Ounces Item Config Object JSON ----------------------- .. versionchanged:: 3.3-s60 attributes, quantity, and barcodeList changed .. literalinclude:: item_config_object.json :language: json Product Feature Attributes ========================== .. todo:: Write about some product attributes affecting features Create or Update Items ====================== .. http:post:: /api-gateway/mdm-service/wms-masterdata/item To create or update a product, an :ref:`Item Object` has to be sent in a HTTP POST request to the above endpoint. The newly created or modified object will be returned in the response if the request is successful. :reqheader Content-Type: the request's content-type has to be passed in this header :reqheader Accept: the response content type depends on :mailheader:`Accept` header :reqheader Authorization: OAuth token to authenticate :statuscode 200: No error :statuscode 400: Bad data .. container:: toggle .. container:: header **Example request**: .. sourcecode:: http POST /api-gateway/mdm-service/wms-masterdata/item HTTP/1.1 Host: example.com Content-Type: application/json Accept: application/json, text/javascript .. literalinclude:: item_object.json :language: JSON .. container:: toggle .. container:: header **Example Response**: .. sourcecode:: http HTTP/1.1 200 OK Content-Type: application/json;charset=UTF-8 .. literalinclude:: item_object.json :language: JSON Search for an Item ================== Products in the system can be searched using their unique attributes. .. http:post:: /api-gateway/mdm-service/wms-masterdata/item/search_v2 For searching, a list of filter objects are given, and the response is a list of :ref:`Item Object`\ s matching the given filters. :reqheader Content-Type: the request's content-type has to be passed in this header :reqheader Accept: the response content type depends on :mailheader:`Accept` header :reqheader Authorization: OAuth token to authenticate :