private RawResult ReadBulkString(byte[] buffer, ref int offset, ref int count) { var prefix = ReadLineTerminatedString(ResultType.Integer, buffer, ref offset, ref count); if (prefix.HasValue) { long i64; if (!prefix.TryGetInt64(out i64)) { throw ExceptionFactory.ConnectionFailure(Multiplexer.IncludeDetailInExceptions, ConnectionFailureType.ProtocolFailure, "Invalid bulk string length", Bridge.ServerEndPoint); } int bodySize = checked ((int)i64); if (bodySize < 0) { return(new RawResult(ResultType.BulkString, null, 0, 0)); } else if (count >= bodySize + 2) { if (buffer[offset + bodySize] != '\r' || buffer[offset + bodySize + 1] != '\n') { throw ExceptionFactory.ConnectionFailure(Multiplexer.IncludeDetailInExceptions, ConnectionFailureType.ProtocolFailure, "Invalid bulk string terminator", Bridge.ServerEndPoint); } var result = new RawResult(ResultType.BulkString, buffer, offset, bodySize); offset += bodySize + 2; count -= bodySize + 2; return(result); } } return(RawResult.Nil); }
internal override bool TryValidate(RawResult result, out bool value) { bool parsed; if (ResultProcessor.DemandZeroOrOneProcessor.TryGet(result, out parsed)) { value = parsed == expectedResult; ConnectionMultiplexer.TraceWithoutContext("exists: " + parsed + "; expected: " + expectedResult + "; voting: " + value); return(true); } value = false; return(false); }
internal override bool TryValidate(RawResult result, out bool value) { switch (result.Type) { case ResultType.BulkString: case ResultType.SimpleString: case ResultType.Integer: var parsed = result.AsRedisValue(); value = (parsed == expectedValue) == expectedEqual; ConnectionMultiplexer.TraceWithoutContext("actual: " + (string)parsed + "; expected: " + (string)expectedValue + "; wanted: " + (expectedEqual ? "==" : "!=") + "; voting: " + value); return(true); } value = false; return(false); }
internal override bool TryValidate(RawResult result, out bool value) { switch (result.Type) { case ResultType.BulkString: case ResultType.SimpleString: case ResultType.Integer: var parsed = result.AsRedisValue(); value = parsed.IsInteger && (expectedLength.CompareTo((long)parsed) == compareToResult); ConnectionMultiplexer.TraceWithoutContext("actual: " + (string)parsed + "; expected: " + expectedLength + "; wanted: " + GetComparisonString() + "; voting: " + value); return(true); } value = false; return(false); }
private RawResult ReadLineTerminatedString(ResultType type, byte[] buffer, ref int offset, ref int count) { int max = offset + count - 2; for (int i = offset; i < max; i++) { if (buffer[i + 1] == '\r' && buffer[i + 2] == '\n') { int len = i - offset + 1; var result = new RawResult(type, buffer, offset, len); count -= (len + 2); offset += (len + 2); return(result); } } return(RawResult.Nil); }
// returns an array of RawResults internal RawResult[] GetArrayOfRawResults() { if (arr == null) { return(null); } else if (arr.Length == 0) { return(new RawResult[0]); } else { var rawResultArray = new RawResult[arr.Length]; for (int i = 0; i < arr.Length; i++) { var rawResult = (RawResult)arr.GetValue(i); rawResultArray.SetValue(rawResult, i); } return(rawResultArray); } }
// internally, this is very similar to RawResult, except it is designed to be usable // outside of the IO-processing pipeline: the buffers are standalone, etc internal static RedisResult TryCreate(PhysicalConnection connection, RawResult result) { try { switch (result.Type) { case ResultType.Integer: case ResultType.SimpleString: case ResultType.BulkString: return(new SingleRedisResult(result.AsRedisValue())); case ResultType.MultiBulk: var items = result.GetItems(); var arr = new RedisResult[items.Length]; for (int i = 0; i < arr.Length; i++) { var next = TryCreate(connection, items[i]); if (next == null) { return(null); // means we didn't understand } arr[i] = next; } return(new ArrayRedisResult(arr)); case ResultType.Error: return(new ErrorRedisResult(result.GetString())); default: return(null); } } catch (Exception ex) { connection?.OnInternalError(ex); return(null); // will be logged as a protocol fail by the processor } }
private RawResult ReadArray(byte[] buffer, ref int offset, ref int count) { var itemCount = ReadLineTerminatedString(ResultType.Integer, buffer, ref offset, ref count); if (itemCount.HasValue) { long i64; if (!itemCount.TryGetInt64(out i64)) { throw ExceptionFactory.ConnectionFailure(Multiplexer.IncludeDetailInExceptions, ConnectionFailureType.ProtocolFailure, "Invalid array length", Bridge.ServerEndPoint); } int itemCountActual = checked ((int)i64); if (itemCountActual < 0) { //for null response by command like EXEC, RESP array: *-1\r\n return(new RawResult(ResultType.SimpleString, null, 0, 0)); } else if (itemCountActual == 0) { //for zero array response by command like SCAN, Resp array: *0\r\n return(RawResult.EmptyArray); } var arr = new RawResult[itemCountActual]; for (int i = 0; i < itemCountActual; i++) { if (!(arr[i] = TryParseResult(buffer, ref offset, ref count)).HasValue) { return(RawResult.Nil); } } return(new RawResult(arr)); } return(RawResult.Nil); }
protected override bool SetResultCore(PhysicalConnection connection, Message message, RawResult result) { switch (result.Type) { case ResultType.BulkString: var raw = result.GetString(); var clients = Parse(raw); SetResult(message, clients); return(true); } return(false); }
protected override bool SetResultCore(PhysicalConnection connection, Message message, RawResult result) { switch (result.Type) { case ResultType.MultiBulk: var arr = result.GetItems(); long i64; if (arr.Length == 2 && arr[1].Type == ResultType.MultiBulk && arr[0].TryGetInt64(out i64)) { var keysResult = new ScanResult(i64, arr[1].GetItemsAsKeys()); SetResult(message, keysResult); return(true); } break; } return(false); }
protected override bool SetResultCore(PhysicalConnection connection, Message message, RawResult result) { if (result.Type == ResultType.SimpleString && result.IsEqual(QUEUED)) { var q = message as QueuedMessage; if (q != null) { q.WasQueued = true; } return(true); } return(false); }
public override bool SetResult(PhysicalConnection connection, Message message, RawResult result) { if (result.IsError) { var tran = message as TransactionMessage; if (tran != null) { string error = result.GetString(); var bridge = connection.Bridge; foreach (var op in tran.InnerOperations) { ServerFail(op.Wrapped, error); bridge.CompleteSyncOrAsync(op.Wrapped); } } } return(base.SetResult(connection, message, result)); }
protected override bool SetResultCore(PhysicalConnection connection, Message message, RawResult result) { var msg = message as ConditionMessage; var condition = msg?.Condition; bool final; if (condition != null && condition.TryValidate(result, out final)) { SetResult(message, final); return(true); } return(false); }
internal abstract bool TryValidate(RawResult result, out bool value);
// true if ready to be completed (i.e. false if re-issued to another server) internal bool ComputeResult(PhysicalConnection connection, RawResult result) { return(resultProcessor == null || resultProcessor.SetResult(connection, this, result)); }
void MatchResult(RawResult result) { // check to see if it could be an out-of-band pubsub message if (connectionType == ConnectionType.Subscription && result.Type == ResultType.MultiBulk) { // out of band message does not match to a queued message var items = result.GetItems(); if (items.Length >= 3 && items[0].IsEqual(message)) { // special-case the configuration change broadcasts (we don't keep that in the usual pub/sub registry) var configChanged = Multiplexer.ConfigurationChangedChannel; if (configChanged != null && items[1].IsEqual(configChanged)) { EndPoint blame = null; try { if (!items[2].IsEqual(RedisLiterals.ByteWildcard)) { blame = Format.TryParseEndPoint(items[2].GetString()); } } catch { /* no biggie */ } Multiplexer.Trace("Configuration changed: " + Format.ToString(blame), physicalName); Multiplexer.ReconfigureIfNeeded(blame, true, "broadcast"); } // invoke the handlers var channel = items[1].AsRedisChannel(ChannelPrefix, RedisChannel.PatternMode.Literal); Multiplexer.Trace("MESSAGE: " + channel, physicalName); if (!channel.IsNull) { Multiplexer.OnMessage(channel, channel, items[2].AsRedisValue()); } return; // AND STOP PROCESSING! } else if (items.Length >= 4 && items[0].IsEqual(pmessage)) { var channel = items[2].AsRedisChannel(ChannelPrefix, RedisChannel.PatternMode.Literal); Multiplexer.Trace("PMESSAGE: " + channel, physicalName); if (!channel.IsNull) { var sub = items[1].AsRedisChannel(ChannelPrefix, RedisChannel.PatternMode.Pattern); Multiplexer.OnMessage(sub, channel, items[3].AsRedisValue()); } return; // AND STOP PROCESSING! } // if it didn't look like "[p]message", then we still need to process the pending queue } Multiplexer.Trace("Matching result...", physicalName); Message msg; lock (outstanding) { Multiplexer.Trace(outstanding.Count == 0, "Nothing to respond to!", physicalName); msg = outstanding.Dequeue(); } Multiplexer.Trace("Response to: " + msg.ToString(), physicalName); if (msg.ComputeResult(this, result)) { Bridge.CompleteSyncOrAsync(msg); } }
protected override bool SetResultCore(PhysicalConnection connection, Message message, RawResult result) { var tran = message as TransactionMessage; if (tran != null) { var bridge = connection.Bridge; var wrapped = tran.InnerOperations; switch (result.Type) { case ResultType.SimpleString: if (tran.IsAborted && result.IsEqual(RedisLiterals.BytesOK)) { connection.Multiplexer.Trace("Acknowledging UNWATCH (aborted electively)"); SetResult(message, false); return(true); } //EXEC returned with a NULL if (!tran.IsAborted && result.IsNull) { connection.Multiplexer.Trace("Server aborted due to failed EXEC"); //cancel the commands in the transaction and mark them as complete with the completion manager foreach (var op in wrapped) { op.Wrapped.Cancel(); bridge.CompleteSyncOrAsync(op.Wrapped); } SetResult(message, false); return(true); } break; case ResultType.MultiBulk: if (!tran.IsAborted) { var arr = result.GetItems(); if (arr == null) { connection.Multiplexer.Trace("Server aborted due to failed WATCH"); foreach (var op in wrapped) { op.Wrapped.Cancel(); bridge.CompleteSyncOrAsync(op.Wrapped); } SetResult(message, false); return(true); } else if (wrapped.Length == arr.Length) { connection.Multiplexer.Trace("Server committed; processing nested replies"); for (int i = 0; i < arr.Length; i++) { if (wrapped[i].Wrapped.ComputeResult(connection, arr[i])) { bridge.CompleteSyncOrAsync(wrapped[i].Wrapped); } } SetResult(message, true); return(true); } } break; } // even if we didn't fully understand the result, we still need to do something with // the pending tasks foreach (var op in wrapped) { op.Wrapped.Fail(ConnectionFailureType.ProtocolFailure, null); bridge.CompleteSyncOrAsync(op.Wrapped); } } return(false); }