コード例 #1
0
        void WriteResponseBackToClient(List <byte> payload)
        {
            _dataRecv += payload.Count();

            if (_tc.Connected)
            {
                try
                {
                    var stream = _tc.GetStream();
                    stream.Write(payload.ToArray(), 0, payload.Count);
                    stream.Flush();
                    ServerComms.LogMessage($"Recieved payload back from Implant (size: {payload.Count} for {_targetId} writing to client");
                    if (ServerComms.IsVerboseOn())
                    {
                        ServerComms.LogMessage($"Wrote {payload.Count} ");
                    }
                    if (!_tc.Connected)
                    {
                        ShutdownClient();
                    }
                }
                catch (Exception ex)
                {
                    ServerComms.LogMessage($"ERROR Writing data back to {ex.Message}");
                    ShutdownClient();
                }
            }
            else
            {
                ShutdownClient();
            }
        }
コード例 #2
0
        public static void NotifyConnection(String target, String status)
        {
            if (ServerComms.IsVerboseOn())
            {
                ServerComms.LogMessage($"Message has arrived back for {target}");
            }

            if (!mapTargetIdToSocksInstance.ContainsKey(target))
            {
                ServerComms.LogError($"Target {target} not found in Socks instance");
                return;
            }
            var socksInstance = mapTargetIdToSocksInstance[target];

            if (status.ToLower() == "open")
            {
                socksInstance._open = true;
            }
            else
            {
                socksInstance._open = false;
            }

            socksInstance.status         = status;
            socksInstance.LastUpdateTime = DateTime.Now;

            if (socksInstance._waitOnConnect)
            {
                socksInstance.SocksTimeout.Set();
            }
        }
コード例 #3
0
ファイル: SocksProxy.cs プロジェクト: y35uishere/SharpSocks
        public static bool ReturnDataCallback(String target, List <byte> payload)
        {
            if (ServerComms.IsVerboseOn())
            {
                ServerComms.LogMessage($"Message has arrived back for {target}");
            }

            if (!mapTargetIdToSocksInstance.ContainsKey(target))
            {
                ServerComms.LogError($"Target {target} not found in Socks instance");
                return(false);
            }

            var socksInstance = mapTargetIdToSocksInstance[target];

            socksInstance.WriteResponseBackToClient(payload);
            return(true);
        }
コード例 #4
0
        void AcceptTcpClient(IAsyncResult iar)
        {
            var tcs = (TcpListener)iar.AsyncState;

            TcpClient tc = null;

            try
            {
                tc = tcs.EndAcceptTcpClient(iar);
            }
            catch (Exception ex)
            {
                if (_serverComms.IsVerboseOn())
                {
                    _serverComms.LogError($"Initial SOCKS Read failed for endpoint {tcs.LocalEndpoint.ToString()} {ex.Message}".Trim());
                }
                return;
            }
            Task.Factory.StartNew(() =>
            {
                try
                {
                    if (ServerComms.IsVerboseOn())
                    {
                        ServerComms.LogMessage($"Message arrived {tcs.LocalEndpoint.ToString()} from {tc.Client.RemoteEndPoint.ToString()}".Trim());
                    }
                    (new SocksProxy()
                    {
                        TOTALSOCKETTIMEOUT = SocketTimeout
                    }).ProcessRequest(tc, WaitOnConnect);
                }
                catch (Exception ex)
                {
                    ServerComms.LogError($"Error occured on EndPoint {tcs.LocalEndpoint.ToString()} shutting down cause of {ex.Message}".Trim());
                    if (tc.Connected)
                    {
                        tc.Close();
                    }
                    return;
                }
            });

            tcs.BeginAcceptTcpClient(AcceptTcpClient, tcs);
        }
コード例 #5
0
        public void ProcessRequest(System.Net.HttpListenerContext ctx)
        {
            System.Net.Cookie sessionCookie = null;
            try
            {
                sessionCookie = ctx.Request.Cookies[_sessionIdName];
                if (sessionCookie == null || String.IsNullOrEmpty(sessionCookie.Value))
                {
                    ctx.Response.StatusCode = 401;
                    ctx.Response.Close();
                    return;
                }
            }
            catch (Exception ex)
            {
                ServerComms.LogMessage($"ERROR Processing session cookie {ex.Message}");
            }

            String decryptedSessionId = null, decryptedStatus = null;

            try
            {
                var decr = _encryption.Decrypt(sessionCookie.Value);
                if (decr == null)
                {
                    throw new Exception($"Can't decrypt session cookie{sessionCookie.Value}");
                }
                var decryptedSessionStatus = UTF8Encoding.UTF8.GetString(decr.ToArray()).Split(':');
                decryptedSessionId = decryptedSessionStatus[0];
                decryptedStatus    = decryptedSessionStatus[1];

                if (String.IsNullOrWhiteSpace(decryptedSessionId))
                {
                    throw new Exception($"Session cookie decrypted to nothing or whitespace");
                }
            }
            catch (Exception ex)
            {
                ServerComms.LogError($"Error occured communicating with implant {ex.Message}");
                ctx.Response.StatusCode = 500;
                ctx.Response.Close();
                return;
            }
            String      response        = null;
            List <byte> responseBytes   = new List <byte>();
            String      uploadedPayload = null;

            //TO DO: Payload is currently coming up as different content types

            try
            {
                if ("POST" == ctx.Request.HttpMethod)
                {
                    //Yeah casting ContentLength64 to an int is not idle, but we should not be uploading anywhere near 2gb+ in one request!!!!!
                    uploadedPayload = (new StreamReader(ctx.Request.InputStream)).ReadToEnd();
                }
                else
                if ("GET" == ctx.Request.HttpMethod)
                {
                    var payloadCookie = ctx.Request.Cookies[PayloadCookieName];
                    //TO DO: Dodgy as hell. Need to make sure this can be tampered/malleable etc
                    //This could be whenever in the request. Need to sort that out
                    if (null != payloadCookie)
                    {
                        if (!String.IsNullOrWhiteSpace(payloadCookie.Value))
                        {
                            uploadedPayload = payloadCookie.Value;
                        }
                    }
                }
            }
            catch (Exception ex)
            {
                ServerComms.LogMessage($"ERROR Processing payload data {decryptedSessionId} {ex.Message}");
            }
            ConnectionDetails dtls = null;

            if (decryptedSessionId == _commandChannel)
            {
                try
                {
                    var   ready = false;
                    Int32 ctr = 0, cmdTasks = 0;

                    ProcessCommandChannelTime();

                    if (null == uploadedPayload || uploadedPayload.Count() == 0)
                    {
                        ServerComms.LogMessage($"Command channel sending {_commandTasks.Count()} tasks ");
                    }
                    else
                    {
                        mapSessionToConnectionDetails[_commandChannel].DataRecv += uploadedPayload.Count();
                        if (_commandTasks.Count() > 0)
                        {
                            ServerComms.LogMessage($"Command channel payload {uploadedPayload.Count()} bytes, sending {_commandTasks.Count()} tasks ");
                        }
                        ProcessCommandChanelImplantMessage(this._encryption.Decrypt(uploadedPayload));
                    }

                    lock (_commandLocker) { cmdTasks = _commandTasks.Count(); }
                    if (cmdTasks == 0)
                    {
                        while (!(ready = _cmdTaskWaitEvent.WaitOne(1000)) && ctr++ < 40)
                        {
                            ;
                        }
                    }

                    lock (_commandLocker) { cmdTasks = _commandTasks.Count(); }
                    if (cmdTasks > 0)
                    {
                        response = new XElement("Response", new XElement("Tasks", PopQueueCommandTasks())).ToString();
                    }
                    else
                    {
                        response = BuildStandardResponse().ToString();
                    }

                    _cmdTaskWaitEvent.Reset();

                    responseBytes.AddRange(UTF8Encoding.UTF8.GetBytes(response));
                    mapSessionToConnectionDetails[_commandChannel].DataSent += responseBytes.Count();
                }
                catch (Exception ex)
                {
                    ServerComms.LogMessage($"ERROR Processing command channel message {ex.Message}");
                }
            }
            else
            {
                try
                {
                    if (decryptedStatus == "closed")
                    {
                        ServerComms.LogMessage($"Close connection has been called on {decryptedSessionId}");
                        //Implant has called time
                        //cleanup the data queue

                        lock (_dataTasks)
                        {
                            if (_dataTasks.ContainsKey(decryptedSessionId))
                            {
                                _dataTasks[decryptedSessionId].Tasks.Clear();
                                var wait = _dataTasks[decryptedSessionId].Wait;
                                _dataTasks[decryptedSessionId].DisposeWait();
                                _dataTasks.Remove(decryptedSessionId);
                            }
                        }

                        lock (_listenerLocker)
                        {
                            _listeners.Remove(decryptedSessionId);
                        }

                        //Let the socks know its over
                        SocksProxy.ImplantCalledClose(decryptedSessionId);
                        ctx.Response.StatusCode = 200;
                        ctx.Response.OutputStream.Close();
                        return;
                    }
                    else if (SocksProxy.IsValidSession(decryptedSessionId))
                    {
                        if (!SocksProxy.IsSessionOpen(decryptedSessionId))
                        {
                            SocksProxy.NotifyConnection(decryptedSessionId, "open");
                        }

                        dtls = SocksProxy.GetDetailsForTargetId(decryptedSessionId);
                        if (null == uploadedPayload || uploadedPayload.Count() == 0)
                        {
                            if (ServerComms.IsVerboseOn())
                            {
                                ServerComms.LogMessage($"Requesting data for connection {dtls.HostPort}:{dtls.Id}");
                            }
                        }
                        else
                        {
                            ServerComms.LogMessage($"[Rx] {dtls.HostPort}:{dtls.Id} {uploadedPayload.Count()} bytes ");
                            SocksProxy.ReturnDataCallback(decryptedSessionId, this._encryption.Decrypt(uploadedPayload));
                        }
                    }
                    else
                    {
                        if (ServerComms.IsVerboseOn())
                        {
                            ServerComms.LogMessage($"Session ID {decryptedSessionId} is not valid");
                        }
                        ctx.Response.StatusCode = 404;
                        ctx.Response.OutputStream.Close();
                        return;
                    }

                    var ctr       = 0;
                    var dataQueue = _dataTasks[decryptedSessionId];
                    var ready     = false;

                    while (null != dataQueue.Wait && !(ready = dataQueue.Wait.WaitOne(1000)) && ctr++ < _longpolltimeout)
                    {
                        ;
                    }

                    if (ready && dataQueue.Tasks.Count() > 0)
                    {
                        lock (dataQueue.PayloadLocker) {
                            while (dataQueue.Tasks.Count != 0)
                            {
                                responseBytes.AddRange(dataQueue.Tasks.Dequeue());
                            }
                        }
                        dataQueue.Wait.Reset();
                        if (null != dtls)
                        {
                            ServerComms.LogMessage($"[Tx] {dtls.HostPort}:{dtls.Id} {responseBytes.Count()} bytes ");
                        }
                    }
                    else
                    if (null != dtls)
                    {
                        ServerComms.LogMessage($"[Tx] {dtls.HostPort}:{dtls.Id} nothing to send. TimedOut: {!ready}");
                    }
                }
                catch (Exception ex)
                {
                    ServerComms.LogMessage($"ERROR Processing response for connection {decryptedSessionId} {ex.Message}");
                }
            }

            try
            {
                ctx.Response.StatusCode = 200;

                var payload = EncryptPayload(responseBytes);
                if (null != payload && payload.Count > 0)
                {
                    ctx.Response.OutputStream.Write(payload.ToArray(), 0, payload.Count());
                }

                ctx.Response.OutputStream.Close();
            }
            catch (Exception ex)
            {
                ServerComms.LogMessage($"ERROR Writing response back to client {ex.Message}");
            }
        }
コード例 #6
0
ファイル: SocksProxy.cs プロジェクト: y35uishere/SharpSocks
        public void ProcessRequest(TcpClient tc, bool waitOnConnect = false)
        {
            _tc = tc;
            var  stream     = _tc.GetStream();
            var  timeout    = 0;
            var  timeoutCtr = 0;
            bool timedOut   = false;

            _waitOnConnect = waitOnConnect;

            if (!stream.CanRead)
            {
                if (ServerComms.IsVerboseOn())
                {
                    ServerComms.LogError($"Something attempted to connect but can't read from the stream");
                }
            }

            while (!timedOut)
            {
                if (!stream.DataAvailable)
                {
                    if (1 == timeoutCtr)
                    {
                        timeout += TIMEBETWEENREADS;
                    }

                    TimeoutEvent.WaitOne(timeout);

                    if (timeoutCtr > (SOCKSCONNECTIONTOREADTIMEOUT / TIMEBETWEENREADS))
                    {
                        //Time out trying to read may as well shutdown the socket
                        ServerComms.LogError($"Timed out ({SOCKSCONNECTIONTOREADTIMEOUT / 1000}s) trying to read from SOCKS conection");
                        status         = "closing (SOCKS time out)";
                        LastUpdateTime = DateTime.Now;
                        return;
                    }
                    timeoutCtr++;
                }
                else
                {
                    var bytesRead   = 0;
                    var lstBuffer   = new List <byte>();
                    var arrayBuffer = new byte[tc.Available];

                    bytesRead = stream.Read(arrayBuffer, 0, tc.Available);
                    lstBuffer.AddRange(arrayBuffer);

                    while (bytesRead > 0 && stream.DataAvailable)
                    {
                        arrayBuffer = new byte[tc.Available];
                        bytesRead  += stream.Read(arrayBuffer, 0, tc.Available);
                        lstBuffer.AddRange(arrayBuffer);
                    }

                    var procResult     = ProcessSocksHeaders(lstBuffer.ToList());
                    var responsePacket = BuildSocks4Response(procResult);

                    stream.Write(responsePacket.ToArray(), 0, responsePacket.Count);
                    stream.Flush();

                    if (Socks4ClientHeader.Socks4ClientHeaderStatus.REQUEST_GRANTED == procResult)
                    {
                        StartCommsWithProxyAndImplant(stream);
                    }
                    else
                    {
                        _tc.Close();
                    }

                    return;
                }
            }
        }