Skip to content

Commit

Permalink
Add more tests and logic fixes
Browse files Browse the repository at this point in the history
  • Loading branch information
mmclure-msft committed Sep 18, 2024
1 parent 3e1084e commit ae21c91
Show file tree
Hide file tree
Showing 2 changed files with 98 additions and 23 deletions.
31 changes: 20 additions & 11 deletions libs/server/Objects/Hash/HashObjectImpl.cs
Original file line number Diff line number Diff line change
Expand Up @@ -519,7 +519,8 @@ private void HashExpire(ref ObjectInput input, ref SpanByteAndMemory output)
}
while (!RespWriteUtils.WriteArrayLength(fieldCount, ref curr, end))
ObjectUtils.ReallocateOutput(ref output, ref isMemory, ref ptr, ref ptrHandle, ref curr, ref end);
for (var fieldIdx = 0; fieldIdx < fieldCount; fieldIdx++)

for (var fieldIdx = 0; fieldIdx < fieldCount; fieldIdx++, currIdx++)
{
var key = parseState.GetArgSliceByRef(currIdx).SpanByte.ToByteArray();

Expand All @@ -528,17 +529,9 @@ private void HashExpire(ref ObjectInput input, ref SpanByteAndMemory output)
{
result = -2;
}
else if (DateTimeOffset.UtcNow >= expiryTime)
{
// If provided expiration time is before or equal to now, delete key
if (hash.Remove(key))
{
this.UpdateSize(key, hashValue.Value, false);
}
result = 2;
}
else
{
Debug.Print($"HashExpire: old: {hashValue.Expiration} new: {expiryTime.Ticks}");
switch (expireOption)
{
case ExpireOption.NX: // Only set if not already set
Expand All @@ -562,7 +555,23 @@ private void HashExpire(ref ObjectInput input, ref SpanByteAndMemory output)
}
if (result != 0) // Option did not reject the operation
{
hashValue.Expiration = expiryTime.Ticks; // Update the expiration time
if (DateTimeOffset.UtcNow >= expiryTime)
{
// If provided expiration time is before or equal to now, delete key
if (hash.Remove(key))
{
this.UpdateSize(key, hashValue.Value, false);
}
result = 2;
Debug.Print($"Deleted value");
}
else
{
hashValue.Expiration = expiryTime.Ticks; // Update the expiration time
hash[key] = hashValue;
Debug.Print($"Set expiration to {expiryTime.Ticks}");
}

}
}
while (!RespWriteUtils.WriteInteger(result, ref curr, end))
Expand Down
90 changes: 78 additions & 12 deletions test/Garnet.test/RespHashTests.cs
Original file line number Diff line number Diff line change
Expand Up @@ -1184,26 +1184,26 @@ public void HExpireCommandParameters(string command)
var lightClientRequest = TestUtils.CreateRequest();
var middleTtl = command switch
{
"HEXPIRE" => 20,
"HPEXPIRE" => 20000,
"HEXPIREAT" => DateTimeOffset.UtcNow.AddSeconds(20).ToUnixTimeSeconds(),
"HPEXPIREAT" => DateTimeOffset.UtcNow.AddSeconds(20).ToUnixTimeMilliseconds(),
"HEXPIRE" => 120,
"HPEXPIRE" => 120000,
"HEXPIREAT" => DateTimeOffset.UtcNow.AddSeconds(120).ToUnixTimeSeconds(),
"HPEXPIREAT" => DateTimeOffset.UtcNow.AddSeconds(120).ToUnixTimeMilliseconds(),
_ => 10
};
var largerTtl = command switch
{
"HEXPIRE" => 30,
"HPEXPIRE" => 30000,
"HEXPIREAT" => DateTimeOffset.UtcNow.AddSeconds(30).ToUnixTimeSeconds(),
"HPEXPIREAT" => DateTimeOffset.UtcNow.AddSeconds(30).ToUnixTimeMilliseconds(),
"HEXPIRE" => 240,
"HPEXPIRE" => 240000,
"HEXPIREAT" => DateTimeOffset.UtcNow.AddSeconds(240).ToUnixTimeSeconds(),
"HPEXPIREAT" => DateTimeOffset.UtcNow.AddSeconds(240).ToUnixTimeMilliseconds(),
_ => 10
};
var smallerTtl = command switch
{
"HEXPIRE" => 10,
"HPEXPIRE" => 10000,
"HEXPIREAT" => DateTimeOffset.UtcNow.AddSeconds(10).ToUnixTimeSeconds(),
"HPEXPIREAT" => DateTimeOffset.UtcNow.AddSeconds(10).ToUnixTimeMilliseconds(),
"HEXPIRE" => 60,
"HPEXPIRE" => 60000,
"HEXPIREAT" => DateTimeOffset.UtcNow.AddSeconds(60).ToUnixTimeSeconds(),
"HPEXPIREAT" => DateTimeOffset.UtcNow.AddSeconds(60).ToUnixTimeMilliseconds(),
_ => 10
};
// This value should ensure the key is deleted
Expand Down Expand Up @@ -1288,6 +1288,72 @@ public void HExpireCommandParameters(string command)
expectedResponse = "$-1\r\n";
actualResponse = Encoding.ASCII.GetString(res).Substring(0, expectedResponse.Length);
ClassicAssert.AreEqual(expectedResponse, actualResponse);

// Add three fields (this also clears their TTLs)
res = lightClientRequest.SendCommand($"HSET {key} field1 1 field2 1 field3 1");
expectedResponse = ":3\r\n";
actualResponse = Encoding.ASCII.GetString(res).Substring(0, expectedResponse.Length);
ClassicAssert.AreEqual(expectedResponse, actualResponse);

// Set TTL on the first two fields
res = lightClientRequest.SendCommand($"{command} {key} {middleTtl} FIELDS 2 field1 field2", 3);
expectedResponse = "*2\r\n:1\r\n:1\r\n";
actualResponse = Encoding.ASCII.GetString(res).Substring(0, expectedResponse.Length);
ClassicAssert.AreEqual(expectedResponse, actualResponse);

// XX on a field with no TTL will fail
res = lightClientRequest.SendCommand($"{command} {key} {middleTtl} XX FIELDS 1 field3", 2);
expectedResponse = "*1\r\n:0\r\n";
actualResponse = Encoding.ASCII.GetString(res).Substring(0, expectedResponse.Length);
ClassicAssert.AreEqual(expectedResponse, actualResponse);

// XX on a field with TTL will succeed
res = lightClientRequest.SendCommand($"{command} {key} {middleTtl} XX FIELDS 1 field1", 2);
expectedResponse = "*1\r\n:1\r\n";
actualResponse = Encoding.ASCII.GetString(res).Substring(0, expectedResponse.Length);
ClassicAssert.AreEqual(expectedResponse, actualResponse);

// NX on a field with TTL will fail
res = lightClientRequest.SendCommand($"{command} {key} {middleTtl} NX FIELDS 1 field1", 2);
expectedResponse = "*1\r\n:0\r\n";
actualResponse = Encoding.ASCII.GetString(res).Substring(0, expectedResponse.Length);
ClassicAssert.AreEqual(expectedResponse, actualResponse);

// NX on a field with no TTL will succeed
res = lightClientRequest.SendCommand($"{command} {key} {middleTtl} NX FIELDS 1 field3", 2);
expectedResponse = "*1\r\n:1\r\n";
actualResponse = Encoding.ASCII.GetString(res).Substring(0, expectedResponse.Length);
ClassicAssert.AreEqual(expectedResponse, actualResponse);

// Set TTL on both fields again
res = lightClientRequest.SendCommand($"{command} {key} {middleTtl} FIELDS 2 field1 field2", 3);
expectedResponse = "*2\r\n:1\r\n:1\r\n";
actualResponse = Encoding.ASCII.GetString(res).Substring(0, expectedResponse.Length);
ClassicAssert.AreEqual(expectedResponse, actualResponse);

// LT when new TTL is larger will fail
res = lightClientRequest.SendCommand($"{command} {key} {largerTtl} LT FIELDS 1 field1", 2);
expectedResponse = "*1\r\n:0\r\n";
actualResponse = Encoding.ASCII.GetString(res).Substring(0, expectedResponse.Length);
ClassicAssert.AreEqual(expectedResponse, actualResponse);

// LT when new TTL is smaller will succeed
res = lightClientRequest.SendCommand($"{command} {key} {smallerTtl} LT FIELDS 1 field1", 2);
expectedResponse = "*1\r\n:1\r\n";
actualResponse = Encoding.ASCII.GetString(res).Substring(0, expectedResponse.Length);
ClassicAssert.AreEqual(expectedResponse, actualResponse);

// GT when new TTL is smaller will fail
res = lightClientRequest.SendCommand($"{command} {key} {smallerTtl} GT FIELDS 1 field2", 2);
expectedResponse = "*1\r\n:0\r\n";
actualResponse = Encoding.ASCII.GetString(res).Substring(0, expectedResponse.Length);
ClassicAssert.AreEqual(expectedResponse, actualResponse);

// GT when new TTL is larger will succeed
res = lightClientRequest.SendCommand($"{command} {key} {largerTtl} GT FIELDS 1 field2", 2);
expectedResponse = "*1\r\n:1\r\n";
actualResponse = Encoding.ASCII.GetString(res).Substring(0, expectedResponse.Length);
ClassicAssert.AreEqual(expectedResponse, actualResponse);
}

#endregion
Expand Down

0 comments on commit ae21c91

Please sign in to comment.