camunda / feel-scala

FEEL parser and interpreter written in Scala
https://camunda.github.io/feel-scala/
Apache License 2.0
120 stars 50 forks source link

New function to deal with timestamp values #663

Open skayliu opened 1 year ago

skayliu commented 1 year ago

Is your feature request related to a problem? Please describe.

When using the mysql connector (related: zeebe-play#196), I got a timestamp value of 1686327521524 or 1686327521, I want to get the datetime of 2023-06-10 00:18:41 or 2023-06-10T00:18:41 so that I can insert the datetime into mysql table.

Describe the solution you'd like New built-in functions to handle Unix timestamps.

Signatures:

to unix timestamp(from: date and time): number

from unix timestamp(from: number): date and time

Examples:

to unix timestamp(@"2023-06-09T16:18:41Z")
// 1686327521524

from unix timestamp(1686327521524)
// "2023-06-09T16:18:41Z"

Related issues

saig0 commented 1 year ago

@skayliu thank you for raising this up. Let's clarify the expectation first.

You need a new built-in function to convert a Unix timestamp into a date-time value. Correct?

Something like:

date and time(1686327521524)
// "2023-06-09T16:18:41Z"

Note that there is an example of how to convert a date-time value into a Unix timestamp here.

We can do the reverse operation with the following expression: (x - the Unix timestamp)

@"1970-01-01T00:00Z" + @"PT1S" * (x / 1000)

This can be a workaround until there is a new function available.

skayliu commented 1 year ago

You need a new built-in function to convert a Unix timestamp into a date-time value. Correct? Something like: date and time(1686327521524) // "2023-06-09T16:18:41Z"

@saig0, Yes, that's what i want.

@"1970-01-01T00:00Z" + @"PT1S" * (x / 1000)

Thank you, i'll give a try.

saig0 commented 1 year ago

@skayliu super. :+1:

I updated the description of the issue and described the new functions to handle Unix timestamps.

saig0 commented 1 year ago

I moved the issue to the backlog because I don't plan to work on this issue soon. There is a workaround available.

The issue is open for contributions. :rocket:

skayliu commented 1 year ago

@saig0 When with the Z will raise a incident ,when without the Z ,it can sucessful insert into mysql table.

Here is my test steps.

MySQL table DDL:

CREATE TABLE `business_process_instance` (
  `id` char(36) COLLATE utf8mb4_unicode_ci NOT NULL,
  `start_time` datetime DEFAULT NULL
  PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;

Modeler: open this bpmn file in Camunda Modeler

<?xml version="1.0" encoding="UTF-8"?>
<bpmn:definitions xmlns:bpmn="http://www.omg.org/spec/BPMN/20100524/MODEL" xmlns:bpmndi="http://www.omg.org/spec/BPMN/20100524/DI" xmlns:dc="http://www.omg.org/spec/DD/20100524/DC" xmlns:zeebe="http://camunda.org/schema/zeebe/1.0" xmlns:di="http://www.omg.org/spec/DD/20100524/DI" xmlns:modeler="http://camunda.org/schema/modeler/1.0" id="Definitions_1oy1tqp" targetNamespace="http://bpmn.io/schema/bpmn" exporter="Camunda Modeler" exporterVersion="5.11.0" modeler:executionPlatform="Camunda Cloud" modeler:executionPlatformVersion="8.2.0">
  <bpmn:process id="Process_02287m3" name="Process_02287m3" isExecutable="true">
    <bpmn:startEvent id="StartEvent_1">
      <bpmn:outgoing>Flow_0msuu6k</bpmn:outgoing>
    </bpmn:startEvent>
    <bpmn:serviceTask id="Activity_0f5kujc" name="Deal With Timestamp" zeebe:modelerTemplate="com.infosys.camundaconnectors.db.mysql.v1" zeebe:modelerTemplateVersion="1" zeebe:modelerTemplateIcon="data:image/svg+xml;utf8,%3Csvg xmlns=&#39;http://www.w3.org/2000/svg&#39; xmlns:xlink=&#39;http://www.w3.org/1999/xlink&#39; width=&#39;18&#39; height=&#39;18&#39; viewBox=&#39;0 0 18 18&#39;%3E%3Cimage xlink:href=&#39;&#39; x=&#39;0&#39; y=&#39;0&#39; height=&#39;100%25&#39; width=&#39;100%25&#39;/%3E%3C/svg%3E">
      <bpmn:extensionElements>
        <zeebe:taskDefinition type="com.infosys.camundaconnectors.db:mysql:1" />
        <zeebe:ioMapping>
          <zeebe:input source="mysql.insert-data" target="operation" />
          <zeebe:input source="127.0.0.1" target="databaseConnection.host" />
          <zeebe:input source="3306" target="databaseConnection.port" />
          <zeebe:input source="root" target="databaseConnection.username" />
          <zeebe:input source="zeebe123" target="databaseConnection.password" />
          <zeebe:input source="exampledb" target="data.databaseName" />
          <zeebe:input source="business_process_instance" target="data.tableName" />
          <zeebe:input source="=[{  &#10;  &#34;id&#34;: processInstance.id,&#10;  &#34;start_time&#34;: @&#34;1970-01-01T00:00&#34; + @&#34;PT1S&#34; * (processInstance.createTime / 1000) + @&#34;PT28800S&#34;&#10;}]" target="data.dataToInsert" />
        </zeebe:ioMapping>
        <zeebe:taskHeaders>
          <zeebe:header key="resultVariable" value="dealStatus" />
        </zeebe:taskHeaders>
      </bpmn:extensionElements>
      <bpmn:incoming>Flow_0msuu6k</bpmn:incoming>
      <bpmn:outgoing>Flow_1ys2snz</bpmn:outgoing>
    </bpmn:serviceTask>
    <bpmn:sequenceFlow id="Flow_0msuu6k" sourceRef="StartEvent_1" targetRef="Activity_0f5kujc" />
    <bpmn:endEvent id="Event_1jic5xr">
      <bpmn:incoming>Flow_1ys2snz</bpmn:incoming>
    </bpmn:endEvent>
    <bpmn:sequenceFlow id="Flow_1ys2snz" sourceRef="Activity_0f5kujc" targetRef="Event_1jic5xr" />
  </bpmn:process>
  <bpmndi:BPMNDiagram id="BPMNDiagram_1">
    <bpmndi:BPMNPlane id="BPMNPlane_1" bpmnElement="Process_02287m3">
      <bpmndi:BPMNShape id="_BPMNShape_StartEvent_2" bpmnElement="StartEvent_1">
        <dc:Bounds x="179" y="99" width="36" height="36" />
      </bpmndi:BPMNShape>
      <bpmndi:BPMNShape id="Activity_0s9c7x9_di" bpmnElement="Activity_0f5kujc">
        <dc:Bounds x="330" y="77" width="100" height="80" />
        <bpmndi:BPMNLabel />
      </bpmndi:BPMNShape>
      <bpmndi:BPMNShape id="Event_1jic5xr_di" bpmnElement="Event_1jic5xr">
        <dc:Bounds x="552" y="99" width="36" height="36" />
      </bpmndi:BPMNShape>
      <bpmndi:BPMNEdge id="Flow_0msuu6k_di" bpmnElement="Flow_0msuu6k">
        <di:waypoint x="215" y="117" />
        <di:waypoint x="330" y="117" />
      </bpmndi:BPMNEdge>
      <bpmndi:BPMNEdge id="Flow_1ys2snz_di" bpmnElement="Flow_1ys2snz">
        <di:waypoint x="430" y="117" />
        <di:waypoint x="552" y="117" />
      </bpmndi:BPMNEdge>
    </bpmndi:BPMNPlane>
  </bpmndi:BPMNDiagram>
</bpmn:definitions>

dataToInsert: The key point here, when with the Z will raise a incident ,when without the Z ,it can sucessful insert into mysql table.

Before:

[{  
  "id": processInstance.id,
  "start_time": @"1970-01-01T00:00Z" + @"PT1S" * (processInstance.createTime / 1000) 
}]

Start current diagram with this payload:

{
    "processInstance":{
        "id":"71182b87-dae4-473a-b378-0907accd9f93",
        "controlState":"active",
        "createTime":1686637501241
    }
}

Variables: In Zeebe-Play we can see variables like this.

processInstance    {"id":"71182b87-dae4-473a-b378-0907accd9f87","controlState":"active","createTime":1686637501241}

data {"dataToInsert":[{"start_time":"2023-06-13T06:25:01Z","id":"71182b87-dae4-473a-b378-0907accd9f87"}],"tableName":"business_process_instance","databaseName":"exampledb"}
operation "mysql.insert-data"

Incidents: And a incident raised.

Unable to insert data into the table: Data truncation: Incorrect datetime value: '2023-06-13T06:25:01Z' for column 'start_time' at row 1

After: dataToInsert: remove the Z and deploy the model again.

[{  
  "id": processInstance.id,
  "start_time": @"1970-01-01T00:00" + @"PT1S" * (processInstance.createTime / 1000) 
}]

Start current diagram with this payload:

{
    "processInstance":{
        "id":"71182b87-dae4-473a-b378-0907accd9f95",
        "controlState":"active",
        "createTime":1686637501241
    }
}

Variables: In Zeebe-Play we can see variables like this and the record inserted successfully

processInstance    {"id":"71182b87-dae4-473a-b378-0907accd9f95","controlState":"active","createTime":1686637501241}
data {"dataToInsert":[{"start_time":"2023-06-13T06:25:01","id":"71182b87-dae4-473a-b378-0907accd9f95"}],"tableName":"business_process_instance","databaseName":"exampledb"}
operation "mysql.insert-data"
dealStatus {"response":"1 row(s) inserted successfully"}

Query mysql table:

SELECT id,start_time FROM business_process_instance WHERE id ='71182b87-dae4-473a-b378-0907accd9f95'

id                                  |start_time         |
------------------------------------+-------------------+
71182b87-dae4-473a-b378-0907accd9f95|2023-06-13 06:25:01|
saig0 commented 1 year ago

@skayliu thank you for sharing. :+1:

It seems that the datetime in MySQL doesn't contain a timezone.