具有表存储的Azure功能-更新实体不起作用

时间:2019-05-23 11:01:21

标签: azure-functions azure-table-storage

我有此代码:

#r "Newtonsoft.Json"
#r "Microsoft.WindowsAzure.Storage"
using System;
using System.IO;
using System.Net.Http;
using Newtonsoft.Json;
using Microsoft.WindowsAzure.Storage.Table;
public async static void Run(string myIoTHubMessage, ICollector<SensorData> tableOutput, CloudTable activityTable,ICollector<SensorData> activityValue, ILogger log)
{
    var data = JsonConvert.DeserializeObject<IoTMessage>(myIoTHubMessage);
    string temp =  data.param2;
    double temperature = double.Parse(temp) * 0.01;

    var sensor = new SensorData { Temperature = temperature.ToString(),DeviceId = data.deviceId, RowKey = data.messageId, PartitionKey = data.deviceId };
    tableOutput.Add(sensor);

    var query = new TableQuery<SensorValue>().Where(TableQuery.GenerateFilterCondition("PartitionKey", QueryComparisons.Equal, data.deviceId.ToString()));
    var segment = await activityTable.ExecuteQuerySegmentedAsync(query, null);

    if (segment.Results.Count == 0)
    {
        activityValue.Add(sensor);
    }
    else
    {
        SensorValue sensorValue = new SensorValue
        {
            PartitionKey = data.deviceId,
            RowKey = data.messageId,
            Temperature = Convert.ToInt16(temperature),
        };

        var operation = TableOperation.Replace(sensorValue);
        await activityTable.ExecuteAsync(operation);
        log.LogInformation(segment.Results.Count.ToString());
    }
}
public class SensorData
{
    public string RowKey { get; set; }
    public string PartitionKey { get; set; }
    public string Temperature { get; set; }
    public string DeviceId {get; set;}
}

public class IoTMessage
{
    public string messageId { get; set; }
    public string temperature { get; set; }
    public string deviceId { get; set; }

}

public class SensorValue : TableEntity
{
    public string RowKey { get; set; }
    public string PartitionKey { get; set; }
    public string Temperature { get; set; }
    public string DeviceId { get; set; }
}

该函数以从传感器获取数据并将其保存到基础的方式工作,但是我想创建第二个表,其中仅保存给定设备的最后一个请求。我检查表中是否存在带有未添加的ID的设备,并且该设备可以正常工作。不幸的是,我在更新给定记录时遇到问题。在我看来像这样:

   if (segment.Results.Count == 0)
        {
            activityValue.Add(sensor);
        }
        else
        {
            SensorValue sensorValue = new SensorValue
            {
                PartitionKey = data.deviceId,
                RowKey = data.messageId,
                Temperature = Convert.ToInt16(temperature),
            };

            var operation = TableOperation.Replace(sensorValue);
            await activityTable.ExecuteAsync(operation);
            log.LogInformation(segment.Results.Count.ToString());
        }

在activityTable表中没有此类设备的情况下进行记录本身不会有任何问题,不幸的是,如果存在此类设备不起作用(即没有错误,但也没有记录更新的情况下进行更新:(

就我而言,在activityTable中,设备ID表中的PartitionKey在请求中发送。我尝试从上午8点开始执行此操作,但是没有任何效果,因此我无法处理:(

1 个答案:

答案 0 :(得分:2)

快速解决方案:将ETag字段设置为*,如您在Azure技巧与窍门文章Updating an item from a Azure Storage Table中所见。

稍长的版本:由于您省略了ETag值,因此根据表存储,您要替换的实体不存在。

  

ETag 属性用于更新期间的乐观并发。这不是时间戳记,因为还有另一个名为 TimeStamp 的属性,用于存储记录的上次更新时间。例如,如果您加载一个实体并想要更新它,则ETag必须与当前存储的匹配。如果您有多个用户编辑同一个项目,而又不想让他们覆盖彼此的更改,那么这对b / c很重要。

将ETag设置为*实际上是将其设置为可以接受任何值的通配符。