# -*- coding: utf-8 -*-
# Copyright (C) 2007 daelstorm. All rights reserved.
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program. If not, see .
#
# Previous copyright below
# Copyright (c) 2003-2004 Hyriand. All rights reserved.
#
# Based on code from PySoulSeek, original copyright note:
# Copyright (c) 2001-2003 Alexander Kanavin. All rights reserved.
"""
This is the actual client code. Actual GUI classes are in the separate modules
"""
from __future__ import division
import time
import datetime
import shutil
from urllib import urlencode
import slskproto
import slskmessages
from slskmessages import newId, PopupMessage, ToBeEncoded
import transfers
import Queue
import threading
from config import *
import string
import types
import locale
import utils
from shares import Shares
from utils import _, CleanFile, findBestEncoding
import os
import logging
from ConfigParser import Error as ConfigParserError
class PeerConnection(object):
"""
Holds information about a peer connection. Not every field may be set
to something. addr is (ip, port) address, conn is a socket object, msgs is
a list of outgoing pending messages, token is a reverse-handshake
number (protocol feature), init is a PeerInit protocol message. (read
slskmessages docstrings for explanation of these)
"""
def __init__(self, addr = None, username = None, conn = None, msgs = None, token = None, init = None, conntimer = None, tryaddr = None, update_callback=None):
self._addr = addr
self.username = username
self._conn = conn
self.msgs = msgs
self._token = token
self.init = init
self.conntimer = conntimer
self.tryaddr = tryaddr
self.update_callback = update_callback
def _get_addr(self):
return self._addr
def _set_addr(self, addr):
self._addr = addr
if self.update_callback:
self.update_callback(self)
addr = property(_get_addr, _set_addr)
def _get_conn(self):
return self._conn
def _set_conn(self, conn):
self._conn = conn
if self.update_callback:
self.update_callback(self)
conn = property(_get_conn, _set_conn)
def _get_token(self):
return self._token
def _set_token(self, token):
self._token = token
if self.update_callback:
self.update_callback(self)
token = property(_get_token, _set_token)
def __repr__(self):
return u"" % (id(self), self.addr, self.username, self.conn, self.token)
class Timeout:
def __init__(self, callback):
self.callback = callback
def timeout(self):
try:
self.callback([self])
except Exception, e:
print("Exception in callback %s: %s" % (self.callback, e))
class ConnectToPeerTimeout(Timeout):
def __init__(self, conn, callback):
self.conn = conn
self.callback = callback
class CloseSearchResultsTimeout(Timeout):
pass
class RespondToDistributedSearchesTimeout(Timeout):
pass
class NetworkEventProcessor:
""" This class contains handlers for various messages from the networking
thread"""
def __init__(self, frame, callback, writelog, setstatus, bindip, configfile):
self.frame = frame
self.callback = callback
self.logMessage = writelog
self.setStatus = setstatus
try:
self.config = Config(configfile)
except ConfigParserError:
#corruptfile = configfile+CleanFile("." + str(datetime.datetime.isoformat(datetime.datetime.now()) + ".corrupt"))
corruptfile = ".".join([configfile, CleanFile(datetime.datetime.now().strftime("%Y-%M-%d_%H:%M:%S")), "corrupt"])
shutil.move(configfile, corruptfile)
short = _("Your config file is corrupt")
long = _("We're sorry, but it seems your configuration file is corrupt. Please reconfigure Nicotine+.\n\nWe renamed your old configuration file to\n%(corrupt)s\nIf you open this file with a text editor you might be able to rescue some of your settings.") % {'corrupt':corruptfile}
log.addwarning(long)
self.config = Config(configfile)
self.callback([PopupMessage(short, long)])
self.bindip = bindip
self.config.frame = frame
self.config.readConfig()
self.peerconns = []
self.peerconns_by_conn = {}
self.peerconns_by_addr = {}
self.peerconns_by_token = {}
self.watchedusers = []
self.ipblock_requested = {}
self.ipignore_requested = {}
self.ip_requested = []
self.PrivateMessageQueue = {}
self.users = {}
self.queue = Queue.Queue(0)
self.shares = Shares(self)
try:
import GeoIP
self.geoip = GeoIP.new(GeoIP.GEOIP_MEMORY_CACHE)
except ImportError:
try:
import _GeoIP
self.geoip = _GeoIP.new(_GeoIP.GEOIP_MEMORY_CACHE)
except ImportError:
self.geoip = None
self.protothread = slskproto.SlskProtoThread(self.frame.networkcallback, self.queue, self.bindip, self.config, self)
uselimit = self.config.sections["transfers"]["uselimit"]
uploadlimit = self.config.sections["transfers"]["uploadlimit"]
limitby = self.config.sections["transfers"]["limitby"]
self.queue.put(slskmessages.SetUploadLimit(uselimit, uploadlimit, limitby))
self.queue.put(slskmessages.SetDownloadLimit( self.config.sections["transfers"]["downloadlimit"]))
if self.config.sections["transfers"]["geoblock"]:
panic = self.config.sections["transfers"]["geopanic"]
cc = self.config.sections["transfers"]["geoblockcc"]
self.queue.put(slskmessages.SetGeoBlock([panic, cc]))
else:
self.queue.put(slskmessages.SetGeoBlock(None))
self.serverconn = None
self.waitport = None
self.chatrooms = None
self.privatechat = None
self.globallist = None
self.userinfo = None
self.userbrowse = None
self.search = None
self.transfers = None
self.userlist = None
self.logintime = None
self.ipaddress = None
self.privileges_left = None
self.servertimer = None
self.servertimeout = -1
self.distribcache = {}
self.branchlevel = 0
self.branchroot = None
self.requestedInfo = {}
self.requestedFolders = {}
self.speed = 0
self.searchResultsConnections = []
searchresultstimeout = CloseSearchResultsTimeout(self.callback)
self.searchResultsTimer = threading.Timer(10, searchresultstimeout.timeout)
self.searchResultsTimer.start()
self.respondDistributed = True
responddistributedtimeout = RespondToDistributedSearchesTimeout(self.callback)
self.respondDistributedTimer = threading.Timer(60, responddistributedtimeout.timeout)
self.respondDistributedTimer.start()
# Callback handlers for messages
self.events = {slskmessages.ConnectToServer:self.ConnectToServer,
slskmessages.ConnectError:self.ConnectError,
slskmessages.IncPort:self.IncPort,
slskmessages.ServerConn:self.ServerConn,
slskmessages.ConnClose:self.ConnClose,
slskmessages.Login:self.Login,
slskmessages.ChangePassword:self.ChangePassword,
slskmessages.MessageUser:self.MessageUser,
slskmessages.PMessageUser:self.PMessageUser,
slskmessages.ExactFileSearch:self.ExactFileSearch,
slskmessages.UserJoinedRoom:self.UserJoinedRoom,
slskmessages.SayChatroom:self.SayChatRoom,
slskmessages.JoinRoom:self.JoinRoom,
slskmessages.UserLeftRoom:self.UserLeftRoom,
slskmessages.QueuedDownloads:self.QueuedDownloads,
slskmessages.GetPeerAddress:self.GetPeerAddress,
slskmessages.OutConn:self.OutConn,
slskmessages.UserInfoReply:self.UserInfoReply,
slskmessages.UserInfoRequest:self.UserInfoRequest,
slskmessages.PierceFireWall:self.PierceFireWall,
slskmessages.CantConnectToPeer:self.CantConnectToPeer,
slskmessages.PeerTransfer:self.PeerTransfer,
slskmessages.SharedFileList:self.SharedFileList,
slskmessages.GetSharedFileList:self.shares.GetSharedFileList,
slskmessages.FileSearchRequest:self.FileSearchRequest,
slskmessages.FileSearchResult:self.FileSearchResult,
slskmessages.ConnectToPeer:self.ConnectToPeer,
slskmessages.GetUserStatus:self.GetUserStatus,
slskmessages.GetUserStats:self.GetUserStats,
slskmessages.Relogged:self.Relogged,
slskmessages.PeerInit:self.PeerInit,
slskmessages.DownloadFile:self.FileDownload,
slskmessages.UploadFile:self.FileUpload,
slskmessages.FileRequest:self.FileRequest,
slskmessages.TransferRequest:self.TransferRequest,
slskmessages.TransferResponse:self.TransferResponse,
slskmessages.QueueUpload:self.QueueUpload,
slskmessages.QueueFailed:self.QueueFailed,
slskmessages.UploadFailed:self.UploadFailed,
slskmessages.PlaceInQueue:self.PlaceInQueue,
slskmessages.FileError:self.FileError,
slskmessages.FolderContentsResponse:self.FolderContentsResponse,
slskmessages.FolderContentsRequest:self.shares.FolderContentsRequest,
slskmessages.RoomList:self.RoomList,
slskmessages.LeaveRoom:self.LeaveRoom,
slskmessages.GlobalUserList:self.GlobalUserList,
slskmessages.AddUser:self.AddUser,
slskmessages.PrivilegedUsers:self.PrivilegedUsers,
slskmessages.AddToPrivileged:self.AddToPrivileged,
slskmessages.CheckPrivileges:self.CheckPrivileges,
slskmessages.ServerPing:self.DummyMessage,
slskmessages.ParentMinSpeed:self.DummyMessage,
slskmessages.ParentSpeedRatio:self.DummyMessage,
slskmessages.Msg85:self.DummyMessage,
slskmessages.Msg12547:self.Msg12547,
slskmessages.ParentInactivityTimeout:self.ParentInactivityTimeout,
slskmessages.SearchInactivityTimeout:self.SearchInactivityTimeout,
slskmessages.MinParentsInCache:self.MinParentsInCache,
slskmessages.Msg89:self.DummyMessage,
slskmessages.WishlistInterval:self.WishlistInterval,
slskmessages.DistribAliveInterval:self.DummyMessage,
slskmessages.ChildDepth: self.ChildDepth,
slskmessages.BranchLevel: self.BranchLevel,
slskmessages.BranchRoot: self.BranchRoot,
slskmessages.DistribChildDepth: self.DistribChildDepth,
slskmessages.DistribBranchLevel: self.DistribBranchLevel,
slskmessages.DistribBranchRoot: self.DistribBranchRoot,
slskmessages.DistribMessage9:self.DistribMessage9,
slskmessages.AdminMessage:self.AdminMessage,
slskmessages.TunneledMessage:self.TunneledMessage,
slskmessages.IncConn:self.IncConn,
slskmessages.PlaceholdUpload:self.PlaceholdUpload,
slskmessages.PlaceInQueueRequest:self.PlaceInQueueRequest,
slskmessages.UploadQueueNotification:self.UploadQueueNotification,
slskmessages.SearchRequest:self.SearchRequest,
slskmessages.FileSearch:self.SearchRequest,
slskmessages.RoomSearch:self.RoomSearchRequest,
slskmessages.UserSearch:self.SearchRequest,
slskmessages.NetInfo:self.NetInfo,
slskmessages.DistribAlive:self.DistribAlive,
slskmessages.DistribSearch:self.DistribSearch,
ConnectToPeerTimeout:self.ConnectToPeerTimeout,
RespondToDistributedSearchesTimeout:self.ToggleRespondDistributed,
CloseSearchResultsTimeout:self.closeSearchResults,
transfers.TransferTimeout:self.TransferTimeout,
slskmessages.RescanShares:self.shares.RescanShares,
slskmessages.RescanBuddyShares:self.shares.RescanBuddyShares,
str:self.Notify,
slskmessages.PopupMessage:self.PopupMessage,
slskmessages.InternalData:self.DisplaySockets,
slskmessages.DebugMessage:self.DebugMessage,
slskmessages.GlobalRecommendations:self.GlobalRecommendations,
slskmessages.Recommendations:self.Recommendations,
slskmessages.ItemRecommendations:self.ItemRecommendations,
slskmessages.SimilarUsers:self.SimilarUsers,
slskmessages.ItemSimilarUsers:self.ItemSimilarUsers,
slskmessages.UserInterests:self.UserInterests,
slskmessages.RoomTickerState:self.RoomTickerState,
slskmessages.RoomTickerAdd:self.RoomTickerAdd,
slskmessages.RoomTickerRemove:self.RoomTickerRemove,
slskmessages.UserPrivileged: self.UserPrivileged,
slskmessages.AckNotifyPrivileges:self.AckNotifyPrivileges,
slskmessages.NotifyPrivileges:self.NotifyPrivileges,
slskmessages.PrivateRoomUsers:self.PrivateRoomUsers,
slskmessages.PrivateRoomOwned:self.PrivateRoomOwned,
slskmessages.PrivateRoomAddUser:self.PrivateRoomAddUser,
slskmessages.PrivateRoomRemoveUser:self.PrivateRoomRemoveUser,
slskmessages.PrivateRoomAdded:self.PrivateRoomAdded,
slskmessages.PrivateRoomRemoved:self.PrivateRoomRemoved,
slskmessages.PrivateRoomDisown:self.PrivateRoomDisown,
slskmessages.PrivateRoomToggle:self.PrivateRoomToggle,
slskmessages.PrivateRoomSomething:self.PrivateRoomSomething,
slskmessages.PrivateRoomOperatorAdded:self.PrivateRoomOperatorAdded,
slskmessages.PrivateRoomOperatorRemoved:self.PrivateRoomOperatorRemoved,
slskmessages.PrivateRoomAddOperator:self.PrivateRoomAddOperator,
slskmessages.PrivateRoomRemoveOperator:self.PrivateRoomRemoveOperator,
slskmessages.PublicRoomMessage:self.PublicRoomMessage,
}
def add_peerconn(self, conn):
print("Adding %s" % repr(conn))
self.peerconns.append(conn)
self._update_peerconn(conn)
def update_peerconn(self, conn):
print("Updating %s" % repr(conn))
self._update_peerconn(conn)
def _update_peerconn(self, conn):
self.peerconns_by_addr[conn.addr] = conn
self.peerconns_by_conn[conn.conn] = conn
self.peerconns_by_token[conn.token] = conn
def get_peerconn_by_addr(self, addr):
if addr is None:
raise KeyError("None shall pass")
peerconn = self.peerconns_by_addr[addr]
if peerconn.addr != addr:
print("Storage mismatch 0: %s != %s" % (peerconn.addr, addr))
raise KeyError('Storage mismatch')
return peerconn
def get_peerconn_by_conn(self, conn):
if conn is None:
raise KeyError("None shall pass")
peerconn = self.peerconns_by_conn[conn]
if peerconn.conn != conn:
print("Storage mismatch 1: %s != %s" % (peerconn.conn, conn))
raise KeyError('Storage mismatch')
return peerconn
def get_peerconn_by_token(self, token):
if token is None:
raise KeyError("None shall pass")
peerconn = self.peerconns_by_token[token]
if peerconn.token != token:
print("Storage mismatch 2: %s != %s" % (peerconn.token, token))
raise KeyError('Storage mismatch')
return peerconn
def del_peerconn(self, conn):
print("Deleting %s" % repr(conn))
#
# Closing sockets causes the following stacktrace:
#
# Traceback (most recent call last):
# File "/usr/lib64/python2.7/threading.py", line 552, in __bootstrap_inner
# self.run()
# File "/home/quinox/chroot/opt/nicotine+/pynicotine/slskproto.py", line 408, in run
# input, output, exc = select.select(conns.keys() + connsinprogress.keys() +[p], connsinprogress.keys() + outsock, [], 0.5)
# File "/usr/lib64/python2.7/socket.py", line 224, in meth
# return getattr(self._sock,name)(*args)
# File "/usr/lib64/python2.7/socket.py", line 170, in _dummy
# raise error(EBADF, 'Bad file descriptor')
# error: [Errno 9] Bad file descriptor
#
# So, let's not close it then.
#try:
# conn.conn.close()
#except AttributeError:
# print("Error closing socket, no socket defined.")
#except Exception, ex:
# print("Error closing socket: %s" % ex)
try:
self.peerconns.remove(conn)
except ValueError:
print("Could not delete from peerconns")
try:
del self.peerconns_by_addr[conn.addr]
except:
print("Failed to delete from peerconns_by_addr")
try:
del self.peerconns_by_conn[conn.conn]
except:
print("Failed to delete from peerconns_by_conn")
try:
del self.peerconns_by_token[conn.token]
except:
print("Failed to delete from peerconns_by_token")
def ProcessRequestToPeer(self, user, message, window = None, address = None):
"""
Sends message to a peer and possibly sets up a window to display
the result.
"""
conn = None
for i in self.peerconns:
if i.username == user and i.init.type == 'P' and message.__class__ is not slskmessages.FileRequest:
conn = i
break
if conn is not None:
if conn.conn is not None:
message.conn = conn.conn
self.queue.put(message)
if window is not None:
window.InitWindow(conn.username, conn.conn)
if message.__class__ is slskmessages.TransferRequest and self.transfers is not None:
self.transfers.gotConnect(message.req, conn.conn)
return
else:
conn.msgs.append(message)
else:
if message.__class__ is slskmessages.FileRequest:
type = 'F'
elif message.__class__ is slskmessages.DistribConn:
type = 'D'
else:
type = 'P'
init = slskmessages.PeerInit(None, self.config.sections["server"]["login"], type, 0)
firewalled = self.config.sections["server"]["firewalled"]
addr = None
behindfw = None
token = None
if user in self.users:
addr = self.users[user].addr
behindfw = self.users[user].behindfw
elif address is not None:
self.users[user] = UserAddr(status = -1, addr=address)
addr = address
if firewalled:
if addr is None:
self.queue.put(slskmessages.GetPeerAddress(user))
elif behindfw is None:
self.queue.put(slskmessages.OutConn(None, addr))
else:
firewalled = 0
if not firewalled:
token = newId()
self.queue.put(slskmessages.ConnectToPeer(token, user, type))
conn = PeerConnection(addr = addr, username = user, msgs = [message], token = token, init = init, update_callback=self.update_peerconn)
self.add_peerconn(conn)
if token is not None:
timeout = 120.0
conntimeout = ConnectToPeerTimeout(conn, self.callback)
timer = threading.Timer(timeout, conntimeout.timeout)
conn.conntimer = timer
timer.start()
if message.__class__ is slskmessages.FileSearchResult:
self.searchResultsConnections.append(conn)
if message.__class__ is slskmessages.TransferRequest and self.transfers is not None:
if conn.addr is None:
self.transfers.gettingAddress(message.req)
elif conn.token is None:
self.transfers.gotAddress(message.req)
else:
self.transfers.gotConnectError(message.req)
def closeSearchResults(self, msg):
if self.searchResultsTimer is not None:
self.searchResultsTimer.cancel()
for conn in self.searchResultsConnections[:]:
self.searchResultsConnections.remove(conn)
self.ClosePeerConnection(conn)
searchresultstimeout = CloseSearchResultsTimeout(self.callback)
self.searchResultsTimer = threading.Timer(10, searchresultstimeout.timeout)
self.searchResultsTimer.start()
def setServerTimer(self):
if self.servertimeout == -1:
self.servertimeout = 15
elif 0 < self.servertimeout < 600:
self.servertimeout = self.servertimeout * 2
self.servertimer = threading.Timer(self.servertimeout, self.ServerTimeout)
self.servertimer.start()
logging.info(_("The server seems to be down or not responding, retrying in %i seconds") %(self.servertimeout))
def ServerTimeout(self):
if self.config.needConfig() <= 1:
self.callback([slskmessages.ConnectToServer()])
def StopTimers(self):
for i in self.peerconns:
if i.conntimer is not None:
i.conntimer.cancel()
if self.servertimer is not None:
self.servertimer.cancel()
if self.searchResultsTimer is not None:
self.searchResultsTimer.cancel()
if self.respondDistributedTimer is not None:
self.respondDistributedTimer.cancel()
if self.transfers is not None:
self.transfers.AbortTransfers()
def ConnectToServer(self, msg):
self.frame.OnConnect(None)
def encodeuser(self, string, user = None):
coding = None
config = self.config.sections
if user and user in config["server"]["userencoding"]:
coding = config["server"]["userencoding"][user]
string = self.decode(string, coding)
try:
return string.encode(locale.nl_langinfo(locale.CODESET))
except:
return string
def encode(self, str, networkenc = None):
if networkenc is None:
networkenc = self.config.sections["server"]["enc"]
if type(str) is types.UnicodeType:
return str.encode(networkenc,'replace')
else:
return str.decode("utf-8",'replace').encode(networkenc,'replace')
def decode(self, string, networkenc = None):
if networkenc is None:
networkenc = self.config.sections["server"]["enc"]
return str(string).decode(networkenc,'replace').encode("utf-8", "replace")
def getencodings(self):
# Encodings and descriptions for ComboBoxes
return [["Latin", 'ascii'], \
["US-Canada", 'cp037'], \
['Hebrew', 'cp424'], \
['US English', 'cp437'], \
['International', 'cp500'], \
['Greek', 'cp737'], \
['Estonian', 'cp775'], \
['Western European', 'cp850'], \
['Central European', 'cp852'], \
['Cyrillic', 'cp855'], \
['Cyrillic', 'cp856'], \
['Turkish', 'cp857'], \
['Portuguese', 'cp860'], \
['Icelandic', 'cp861'],\
['Hebrew', 'cp862'], \
['French Canadian', 'cp863'], \
['Arabic', 'cp864'], \
['Nordic', 'cp865'], \
['Cyrillic', 'cp866'], \
['Latin-9', 'cp869'], \
['Thai', 'cp874'], \
['Greek', 'cp875'],\
['Japanese', 'cp932'], \
['Chinese Simple', 'cp936'], \
['Korean', 'cp949'], \
['Chinese Traditional', 'cp950'], \
['Urdu', 'cp1006'], \
['Turkish', 'cp1026'], \
['Latin', 'cp1140'], \
['Central European', 'cp1250'], \
['Cyrillic', 'cp1251'], \
['Latin', 'cp1252'], \
['Greek', 'cp1253'], \
['Turkish', 'cp1254'], \
['Hebrew', 'cp1255'], \
['Arabic', 'cp1256'], \
['Baltic', 'cp1257'], \
['Vietnamese', 'cp1258'], \
['Latin', 'iso8859-1'], \
['Latin 2', 'iso8859-2'], \
['South European', 'iso8859-3'], \
['North European', 'iso8859-4'], \
['Cyrillic', 'iso8859-5'], \
['Arabic', 'iso8859-6'], \
['Greek', 'iso8859-7'], \
['Hebrew', 'iso8859-8'], \
['Turkish', 'iso8859-9'], \
['Nordic', 'iso8859-10'], \
['Thai', 'iso8859-11'], \
['Baltic', 'iso8859-13'], \
['Celtic', 'iso8859-14'], \
['Western European', 'iso8859-15'], \
['South-Eastern European', 'iso8859-16'], \
['Cyrillic', 'koi8-r'], \
['Latin', 'latin-1'], \
['Japanese', 'shift_jis'], \
['Korean', 'euc_kr'], \
['Cyrillic', 'mac-cyrillic'], \
['Greek', 'mac-greek'], \
['Icelandic', 'mac-iceland'], \
['Latin 2', 'mac-latin2'], \
['Latin', 'mac-roman'], \
['Turkish', 'mac-turkish'], \
['International', 'utf-16'], \
['International', 'utf-7'], \
['International', 'utf-8']]
## Notify user of error when recieving or sending a message
# @param self NetworkEventProcessor (Class)
# @param string a string containing an error message
def Notify(self, string):
self.logMessage("%s" % self.decode(string))
def PopupMessage(self, msg):
self.setStatus(_(msg.title))
self.frame.PopupMessage(msg)
def DebugMessage(self, msg):
self.logMessage(msg.msg, msg.debugLevel)
def DisplaySockets(self, msg):
self.frame.SetSocketStatus(msg.msg)
def ConnectError(self, msg):
if msg.connobj.__class__ is slskmessages.ServerConn:
self.setStatus(_("Can't connect to server %(host)s:%(port)s: %(error)s") % {'host': msg.connobj.addr[0], 'port': msg.connobj.addr[1], 'error': self.decode(msg.err) } )
if msg.connobj.addr[1] == 2240:
self.logMessage(_("WARNING: You tried to connect to the old server, which might have been shut down. To connect to the new Soulseek server edit your settings and change the server port from 2240 to 2242"))
self.setServerTimer()
if self.serverconn is not None:
self.serverconn = None
self.frame.ConnectError(msg)
elif msg.connobj.__class__ is slskmessages.OutConn:
for i in self.peerconns[:]:
if i.addr == msg.connobj.addr and i.conn is None:
if i.token is None:
i.token = newId()
self.queue.put(slskmessages.ConnectToPeer(i.token, i.username, i.init.type))
if i.username in self.users:
self.users[i.username].behindfw = "yes"
for j in i.msgs:
if j.__class__ is slskmessages.TransferRequest and self.transfers is not None:
self.transfers.gotConnectError(j.req)
conntimeout = ConnectToPeerTimeout(i, self.callback)
timer = threading.Timer(120.0, conntimeout.timeout)
timer.start()
if i.conntimer is not None:
i.conntimer.cancel()
i.conntimer = timer
else:
for j in i.msgs:
if j.__class__ in [slskmessages.TransferRequest, slskmessages.FileRequest] and self.transfers is not None:
self.transfers.gotCantConnect(j.req)
self.logMessage(_("Can't connect to %s, sending notification via the server") %(i.username), 3)
self.queue.put(slskmessages.CantConnectToPeer(i.token, i.username))
if i.conntimer is not None:
i.conntimer.cancel()
self.del_peerconn(i)
break
else:
self.logMessage("%s %s %s" %(msg.err, msg.__class__, vars(msg)), 4)
else:
self.logMessage("%s %s %s" %(msg.err, msg.__class__, vars(msg)), 4)
self.ClosedConnection(msg.connobj.conn, msg.connobj.addr)
def IncPort(self, msg):
self.waitport = msg.port
self.setStatus(_("Listening on port %i") %(msg.port))
#print "Now reporting %s to the server" % (msg.port,)
def ServerConn(self, msg):
self.setStatus(_("Connected to server %(host)s:%(port)s, logging in...") % {'host':msg.addr[0], 'port': msg.addr[1]})
if msg.addr[1] == 2240:
self.logMessage(_("WARNING: You are connected to the old server, which will most likely be shut down in the near future. To connect to the new Soulseek server edit your settings and change the server port from 2240 to 2242"))
time.sleep(1)
self.serverconn = msg.conn
self.servertimeout = -1
self.users = {}
self.queue.put(slskmessages.Login(self.config.sections["server"]["login"], self.config.sections["server"]["passw"],157)) #155, 156, 157, 180
if self.waitport is not None:
self.queue.put(slskmessages.SetWaitPort(self.waitport))
def PeerInit(self, msg):
conn = PeerConnection(addr = msg.conn.addr, username = msg.user, conn = msg.conn.conn, init = msg, msgs = [], update_callback=self.update_peerconn)
self.add_peerconn(conn)
def ConnClose(self, msg):
self.ClosedConnection(msg.conn, msg.addr)
def ClosedConnection(self, conn, addr):
if conn == self.serverconn:
self.setStatus(_("Disconnected from server %(host)s:%(port)s") %{'host':addr[0], 'port':addr[1]})
userchoice = bool(self.frame.manualdisconnect)
if not self.frame.manualdisconnect:
self.setServerTimer()
else:
self.frame.manualdisconnect = 0
if self.searchResultsTimer is not None:
self.searchResultsTimer.cancel()
if self.respondDistributedTimer is not None:
self.respondDistributedTimer.cancel()
self.serverconn = None
self.watchedusers = []
if self.transfers is not None:
self.transfers.AbortTransfers()
self.transfers.SaveDownloads()
self.privatechat = self.chatrooms = self.userinfo = self.userbrowse = self.search = self.transfers = self.userlist = None
self.frame.ConnClose(conn, addr)
self.frame.pluginhandler.ServerDisconnectNotification(userchoice)
else:
try:
i = self.get_peerconn_by_conn(conn)
except KeyError:
# This seem to happen persistenly with some clients. Are they at fault?
self.logMessage(_("Removed connection closed by peer: %(conn_obj)s %(address)s") %{'conn_obj':conn, 'address':addr}, 3)
self.queue.put(slskmessages.ConnClose(conn))
return
self.logMessage(_("Connection closed by peer: %s") % self.decode(vars(i)), 3)
if i.conntimer is not None:
i.conntimer.cancel()
if self.transfers is not None:
self.transfers.ConnClose(conn, addr)
if i == self.GetDistribConn():
self.DistribConnClosed(i)
self.del_peerconn(i)
def Login(self, msg):
self.logintime = time.time()
conf = self.config.sections
if msg.success:
self.setStatus(_("Logged in, getting the list of rooms..."))
self.transfers = transfers.Transfers(conf["transfers"]["downloads"], self.queue, self, self.users)
if msg.ip is not None:
self.ipaddress = msg.ip
self.privatechat, self.chatrooms, self.userinfo, self.userbrowse, self.search, downloads, uploads, self.userlist = self.frame.InitInterface(msg)
self.transfers.setTransferPanels(downloads, uploads)
self.shares.sendNumSharedFoldersFiles()
self.queue.put(slskmessages.SetStatus((not self.frame.away)+1))
for thing in self.config.sections["interests"]["likes"]:
self.queue.put(slskmessages.AddThingILike(self.encode(thing)))
for thing in self.config.sections["interests"]["dislikes"]:
self.queue.put(slskmessages.AddThingIHate(self.encode(thing)))
if not len(self.distribcache):
self.queue.put(slskmessages.HaveNoParent(1))
self.queue.put(slskmessages.NotifyPrivileges(1, self.config.sections["server"]["login"]))
self.privatechat.Login()
self.queue.put(slskmessages.CheckPrivileges())
self.queue.put(slskmessages.PrivateRoomToggle(self.config.sections["server"]["private_chatrooms"]))
else:
self.frame.manualdisconnect = 1
self.setStatus(_("Can not log in, reason: %s") %(msg.reason))
self.logMessage(_("Can not log in, reason: %s") %(msg.reason))
self.frame.settingswindow.SetSettings(self.config.sections)
self.frame.settingswindow.SwitchToPage("Server")
#self.frame.settingswindow.Hilight(self.frame.settingswindow.pages["Server"].Login)
self.frame.settingswindow.Hilight(self.frame.settingswindow.pages["Server"].Password)
def ChangePassword(self, msg):
password = msg.password
self.config.sections["server"]["passw"] = password
self.config.writeConfig()
self.frame.PopupMessage2( _("Your password has been changed"), "Password is %s" % password)
def NotifyPrivileges(self, msg):
if msg.token != None:
pass
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def UserPrivileged(self, msg):
if self.transfers is not None:
if msg.privileged is True:
self.transfers.addToPrivileged(msg.user)
def AckNotifyPrivileges(self, msg):
if msg.token != None:
# Until I know the syntax, sending this message is probably a bad idea
self.queue.put(slskmessages.AckNotifyPrivileges(msg.token))
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def PMessageUser(self, msg):
user = None
ip = None
port = None
# Get peer's username, ip and port
try:
i = self.get_peerconn_by_conn(msg.conn.conn)
user = i.username
if i.addr is not None:
ip, port = i.addr
except KeyError:
print("Found nothing: %s" % msg.conn.conn)
pass
if user == None:
# No peer connection
return
if user != msg.user:
text = _("(Warning: %(realuser)s is attempting to spoof %(fakeuser)s) ") % {"realuser": user, "fakeuser": msg.user} + msg.msg
msg.user = user
else:
text = msg.msg
if self.privatechat is not None:
self.privatechat.ShowMessage(msg, text, status=0)
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def MessageUser(self, msg):
if msg.user in self.config.sections["server"]["userencoding"]:
encodings = [self.config.sections["server"]["userencoding"][msg.user]] + self.config.sections["server"]["fallbackencodings"]
encodings.append(self.config.sections["server"]["enc"])
else:
encodings = [self.config.sections["server"]["enc"]] + self.config.sections["server"]["fallbackencodings"]
msg.msg = findBestEncoding(msg.msg, encodings)
status = 0
if self.logintime:
if time.time() <= self.logintime + 2:
# Offline message
status = 1
if self.privatechat is not None:
tuple = self.frame.pluginhandler.IncomingPrivateChatEvent(msg.user, msg.msg)
if tuple != None:
(u, msg.msg) = tuple
self.privatechat.ShowMessage(msg, msg.msg, status=status)
self.frame.pluginhandler.IncomingPrivateChatNotification(msg.user, msg.msg)
self.queue.put(slskmessages.MessageAcked(msg.msgid))
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def UserJoinedRoom(self, msg):
if self.chatrooms is not None:
self.chatrooms.roomsctrl.UserJoinedRoom(msg)
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def PublicRoomMessage(self, msg):
msg.msg = findBestEncoding(msg.msg, [self.config.sections["server"]["enc"]] + self.config.sections["server"]["fallbackencodings"])
if self.chatrooms is not None:
self.chatrooms.roomsctrl.PublicRoomMessage(msg, msg.msg)
else:
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def JoinRoom(self, msg):
if self.chatrooms is not None:
self.chatrooms.roomsctrl.JoinRoom(msg)
ticker = ""
if msg.room in self.config.sections["ticker"]["rooms"]:
ticker = self.config.sections["ticker"]["rooms"][msg.room]
elif self.config.sections["ticker"]["default"]:
ticker = self.config.sections["ticker"]["default"]
if ticker:
encoding = self.config.sections["server"]["enc"]
if msg.room in self.config.sections["server"]["roomencoding"]:
encoding = self.config.sections["server"]["roomencoding"][msg.room]
self.queue.put(slskmessages.RoomTickerSet(msg.room, ToBeEncoded(ticker, encoding)))
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def PrivateRoomUsers(self, msg):
if self.chatrooms is not None:
self.chatrooms.roomsctrl.PrivateRoomUsers(msg)
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def PrivateRoomOwned(self, msg):
if self.chatrooms is not None:
self.chatrooms.roomsctrl.PrivateRoomOwned(msg)
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def PrivateRoomAddUser(self, msg):
if self.chatrooms is not None:
self.chatrooms.roomsctrl.PrivateRoomAddUser(msg)
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def PrivateRoomRemoveUser(self, msg):
if self.chatrooms is not None:
self.chatrooms.roomsctrl.PrivateRoomRemoveUser(msg)
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def PrivateRoomOperatorAdded(self, msg):
if self.chatrooms is not None:
self.chatrooms.roomsctrl.PrivateRoomOperatorAdded(msg)
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def PrivateRoomOperatorRemoved(self, msg):
if self.chatrooms is not None:
self.chatrooms.roomsctrl.PrivateRoomOperatorRemoved(msg)
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def PrivateRoomAddOperator(self, msg):
if self.chatrooms is not None:
self.chatrooms.roomsctrl.PrivateRoomAddOperator(msg)
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def PrivateRoomRemoveOperator(self, msg):
if self.chatrooms is not None:
self.chatrooms.roomsctrl.PrivateRoomRemoveOperator(msg)
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def PrivateRoomAdded(self, msg):
if self.chatrooms is not None:
self.chatrooms.roomsctrl.PrivateRoomAdded(msg)
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def PrivateRoomRemoved(self, msg):
if self.chatrooms is not None:
self.chatrooms.roomsctrl.PrivateRoomRemoved(msg)
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def PrivateRoomDisown(self, msg):
if self.chatrooms is not None:
self.chatrooms.roomsctrl.PrivateRoomDisown(msg)
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def PrivateRoomToggle(self, msg):
if self.chatrooms is not None:
self.chatrooms.roomsctrl.TogglePrivateRooms(msg.enabled)
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def PrivateRoomSomething(self, msg):
#msg.debug()
pass
def LeaveRoom(self, msg):
if self.chatrooms is not None:
self.chatrooms.roomsctrl.LeaveRoom(msg)
else:
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def PrivateMessageQueueAdd(self, msg, text):
user = msg.user
if user not in self.PrivateMessageQueue:
self.PrivateMessageQueue[user] = [[msg, text]]
else:
self.PrivateMessageQueue[user].append([msg, text])
def PrivateMessageQueueProcess(self, user):
if user in self.PrivateMessageQueue:
for data in self.PrivateMessageQueue[user][:]:
msg, text = data
self.PrivateMessageQueue[user].remove(data)
self.privatechat.ShowMessage(msg, text, status=0)
def ipIgnored(self, address):
if address is None:
return True
ips = self.config.sections["server"]["ipignorelist"]
s_address = address.split(".")
for ip in ips:
# No Wildcard in IP
if "*" not in ip:
if address == ip:
return True
continue
# Wildcard in IP
parts = ip.split(".")
seg = 0
for part in parts:
# Stop if there's no wildcard or matching string number
if part not in ( s_address[seg], "*"):
break
seg += 1
# Last time around
if seg == 4:
# Wildcard blocked
return True
# Not blocked
return False
def SayChatRoom(self, msg):
if msg.room in self.config.sections["server"]["roomencoding"]:
encodings = [self.config.sections["server"]["roomencoding"][msg.room]] + self.config.sections["server"]["fallbackencodings"]
encodings.append(self.config.sections["server"]["enc"])
else:
encodings = [self.config.sections["server"]["enc"]] + self.config.sections["server"]["fallbackencodings"]
msg.msg = findBestEncoding(msg.msg, encodings)
if self.chatrooms is not None:
event = self.frame.pluginhandler.IncomingPublicChatEvent(msg.room, msg.user, msg.msg)
if event != None:
(r, n, msg.msg) = event
self.chatrooms.roomsctrl.SayChatRoom(msg, msg.msg)
self.frame.pluginhandler.IncomingPublicChatNotification(msg.room, msg.user, msg.msg)
else:
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def AddUser(self, msg):
if msg.user not in self.watchedusers:
self.watchedusers.append(msg.user)
if not msg.userexists:
if msg.user not in self.users:
self.users[msg.user] = UserAddr(status = -1)
if self.search is not None:
self.search.NonExistantUser(msg.user)
if self.transfers is not None:
self.transfers.getAddUser(msg)
else:
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
if msg.status is not None:
self.GetUserStatus(msg)
elif msg.userexists and msg.status is None:
self.queue.put(slskmessages.GetUserStatus(msg.user))
if msg.files is not None:
self.GetUserStats(msg)
elif msg.userexists and msg.files is None:
self.queue.put(slskmessages.GetUserStats(msg.user))
def PrivilegedUsers(self, msg):
if self.transfers is not None:
self.transfers.setPrivilegedUsers(msg.users)
self.logMessage(_("%i privileged users") %(len(msg.users)))
self.queue.put(slskmessages.HaveNoParent(1))
self.queue.put(slskmessages.GetUserStats(self.config.sections["server"]["login"]))
if sys.platform == 'darwin':
self.logMessage("If you like using Nicotine+ on OS X, please consider sending a postcard to the maintainer, here's the address:\nAlexander Kanavin\nVetehisenkuja 4 B29\n00530 Helsinki Finland")
self.frame.pluginhandler.ServerConnectNotification()
else:
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def AddToPrivileged(self, msg):
if self.transfers is not None:
self.transfers.addToPrivileged(msg.user)
#self.logMessage(_("User %s added to privileged list") %(msg.user), 1)
else:
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def CheckPrivileges(self, msg):
mins = msg.seconds // 60
hours = mins // 60
days = hours // 24
wheretogetthem = " " + _("You can acquire privileges by donating at %(url)s") % {'url':'http://www.slsknet.org/userlogin.php?' + urlencode({'username':self.config.sections["server"]["login"]})}
if msg.seconds == 0:
self.logMessage(_("You have no privileges left. They are not needed for Nicotine+ to function properly.") + wheretogetthem)
else:
self.logMessage(_("%(days)i days, %(hours)i hours, %(minutes)i minutes, %(seconds)i seconds of download privileges left.") % {'days':days, 'hours':hours % 24, 'minutes':mins % 60, 'seconds':msg.seconds % 60} + wheretogetthem)
self.privileges_left = msg.seconds
def AdminMessage(self, msg):
self.logMessage("%s" %(msg.msg))
def ChildDepth(self, msg):
# Todo: Distributed search messages need to implemented
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def BranchLevel(self, msg):
# Todo: Distributed search messages need to implemented
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def BranchRoot(self, msg):
# Todo: Distributed search messages need to implemented
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def DistribChildDepth(self, msg):
# Todo: Distributed search messages need to implemented
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def DistribBranchLevel(self, msg):
# Todo: Distributed search messages need to implemented
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def DistribBranchRoot(self, msg):
# Todo: Distributed search messages need to implemented
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def DistribMessage9(self, msg):
# Todo: Distributed search messages need to implemented
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def DummyMessage(self, msg):
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def Msg12547(self, msg):
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def ParentInactivityTimeout(self, msg):
pass
#msg.debug()
def SearchInactivityTimeout(self, msg):
pass
#msg.debug()
def MinParentsInCache(self, msg):
pass
#msg.debug()
def WishlistInterval(self, msg):
if self.search is not None:
self.search.SetInterval(msg)
else:
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def GetUserStatus(self, msg):
# Causes recursive requests when privileged?
#self.queue.put(slskmessages.AddUser(msg.user))
if msg.user in self.users:
if msg.status == 0:
self.users[msg.user] = UserAddr(status = msg.status)
else:
self.users[msg.user].status = msg.status
else:
self.users[msg.user] = UserAddr(status = msg.status)
if msg.privileged != None:
if msg.privileged == 1:
if self.transfers is not None:
self.transfers.addToPrivileged(msg.user)
else:
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
self.frame.GetUserStatus(msg)
if self.userlist is not None:
self.userlist.GetUserStatus(msg)
if self.transfers is not None:
self.transfers.GetUserStatus(msg)
if self.privatechat is not None:
self.privatechat.GetUserStatus(msg)
if self.userinfo is not None:
self.userinfo.GetUserStatus(msg)
if self.userbrowse is not None:
self.userbrowse.GetUserStatus(msg)
if self.search is not None:
self.search.GetUserStatus(msg)
if self.chatrooms is not None:
self.chatrooms.roomsctrl.GetUserStatus(msg)
else:
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def UserInterests(self, msg):
if self.userinfo is not None:
self.userinfo.ShowInterests(msg)
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def GetUserStats(self, msg):
if msg.user == self.config.sections["server"]["login"]:
self.speed = msg.avgspeed
self.frame.GetUserStats(msg)
if self.chatrooms is not None:
self.chatrooms.roomsctrl.GetUserStats(msg)
if self.userinfo is not None:
self.userinfo.GetUserStats(msg)
if self.userlist is not None:
self.userlist.GetUserStats(msg)
else:
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
stats = {
'avgspeed': msg.avgspeed,
'downloadnum': msg.downloadnum,
'files': msg.files,
'dirs': msg.dirs,
}
self.frame.pluginhandler.UserStatsNotification(msg.user, stats)
def UserLeftRoom(self, msg):
if self.chatrooms is not None:
self.chatrooms.roomsctrl.UserLeftRoom(msg)
else:
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def QueuedDownloads(self, msg):
# if self.chatrooms is not None:
# self.chatrooms.roomsctrl.QueuedDownloads(msg)
# else:
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def GetPeerAddress(self, msg):
for i in self.peerconns:
if i.username == msg.user and i.addr is None:
if msg.port != 0 or i.tryaddr == 10:
if i.tryaddr == 10:
self.logMessage(_("Server reported port 0 for the 10th time for user %(user)s, giving up") %{'user':msg.user}, 3)
elif i.tryaddr is not None:
self.logMessage(_("Server reported non-zero port for user %(user)s after %(tries)i retries") %{'user':msg.user, 'tries':i.tryaddr}, 3)
i.addr = (msg.ip, msg.port)
i.tryaddr = None
self.queue.put(slskmessages.OutConn(None, i.addr))
for j in i.msgs:
if j.__class__ is slskmessages.TransferRequest and self.transfers is not None:
self.transfers.gotAddress(j.req)
else:
if i.tryaddr is None:
i.tryaddr = 1
self.logMessage(_("Server reported port 0 for user %(user)s, retrying") %{'user':msg.user}, 3)
else:
i.tryaddr +=1
self.queue.put(slskmessages.GetPeerAddress(msg.user))
break
else:
if msg.user in self.users:
self.users[msg.user].addr = (msg.ip, msg.port)
else:
self.users[msg.user] = UserAddr(addr = (msg.ip, msg.port) )
if msg.user in self.ipblock_requested:
if self.ipblock_requested[msg.user]:
self.frame.OnUnBlockUser(msg.user)
else:
self.frame.OnBlockUser(msg.user)
del self.ipblock_requested[msg.user]
return
if msg.user in self.ipignore_requested:
if self.ipignore_requested[msg.user]:
self.frame.OnUnIgnoreUser(msg.user)
else:
self.frame.OnIgnoreUser(msg.user)
del self.ipignore_requested[msg.user]
return
# From this point on all paths should call
# self.frame.pluginhandler.UserResolveNotification precisely once
if msg.user in self.PrivateMessageQueue:
self.PrivateMessageQueueProcess(msg.user)
if msg.user not in self.ip_requested:
self.frame.pluginhandler.UserResolveNotification(msg.user, msg.ip, msg.port)
return
self.ip_requested.remove(msg.user)
import socket
if self.geoip:
cc = self.geoip.country_name_by_addr(msg.ip)
cn = self.geoip.country_code_by_addr(msg.ip)
if cn is not None:
self.frame.HasUserFlag(msg.user, "flag_"+cn)
else:
cc = ""
if cc:
cc = " (%s)" % cc
else:
cc = ""
try:
hostname = socket.gethostbyaddr(msg.ip)[0]
message = _("IP address of %(user)s is %(ip)s, name %(host)s, port %(port)i%(country)s") %{'user':msg.user, 'ip':msg.ip, 'host':hostname, 'port':msg.port, 'country':cc}
except:
message = _("IP address of %(user)s is %(ip)s, port %(port)i%(country)s") %{'user':msg.user, 'ip':msg.ip, 'port':msg.port, 'country':cc}
self.logMessage(message)
self.frame.pluginhandler.UserResolveNotification(msg.user, msg.ip, msg.port, cc)
def Relogged(self, msg):
self.logMessage(_("Someone else is logging in with the same nickname, server is going to disconnect us"))
self.frame.manualdisconnect = 1
self.frame.pluginhandler.ServerDisconnectNotification(False)
def OutConn(self, msg):
try:
i = self.get_peerconn_by_addr(msg.addr)
except KeyError:
print("What am I supposed to do? %s" % msg.addr)
return
if i.conn is None:
if i.token is None:
i.init.conn = msg.conn
self.queue.put(i.init)
else:
self.queue.put(slskmessages.PierceFireWall(msg.conn, i.token))
i.conn = msg.conn
for j in i.msgs:
if j.__class__ is slskmessages.UserInfoRequest and self.userinfo is not None:
self.userinfo.InitWindow(i.username, msg.conn)
if j.__class__ is slskmessages.GetSharedFileList and self.userbrowse is not None:
self.userbrowse.InitWindow(i.username, msg.conn)
if j.__class__ is slskmessages.FileRequest and self.transfers is not None:
self.transfers.gotFileConnect(j.req, msg.conn)
if j.__class__ is slskmessages.TransferRequest and self.transfers is not None:
self.transfers.gotConnect(j.req, msg.conn)
j.conn = msg.conn
self.queue.put(j)
i.msgs = []
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 3)
def IncConn(self, msg):
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 3)
def ConnectToPeer(self, msg):
init = slskmessages.PeerInit(None, msg.user, msg.type, 0)
self.queue.put(slskmessages.OutConn(None, (msg.ip, msg.port), init))
conn = PeerConnection(addr = (msg.ip, msg.port), username = msg.user, msgs = [], token = msg.token, init = init, update_callback=self.update_peerconn)
self.add_peerconn(conn)
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 3)
def CheckUser(self, user, addr):
"""
Check if this user is banned, geoip-blocked, and which shares
it is allowed to access based on transfer and shares settings.
"""
if user in self.config.sections["server"]["banlist"]:
if self.config.sections["transfers"]["usecustomban"]:
return 0, _("Banned (%s)") % self.config.sections["transfers"]["customban"]
else:
return 0, _("Banned")
if user in [i[0] for i in self.config.sections["server"]["userlist"]] and self.config.sections["transfers"]["enablebuddyshares"]:
# For sending buddy-only shares
return 2, ""
if user in [i[0] for i in self.config.sections["server"]["userlist"]]:
return 1, ""
if self.config.sections["transfers"]["friendsonly"]:
return 0, _("Sorry, friends only")
if not self.geoip or not self.config.sections["transfers"]["geoblock"]:
return 1, _("geoip")
cc = None
if addr is not None:
cc = self.geoip.country_code_by_addr(addr)
if not cc:
if self.config.sections["transfers"]["geopanic"]:
return 0, _("Sorry, geographical paranoia")
else:
return 1, ""
if self.config.sections["transfers"]["geoblockcc"][0].find(cc) >= 0:
return 0, _("Sorry, your country is blocked")
return 1, ""
def CheckSpoof(self, user, ip, port):
if user not in self.users:
return 0
if self.users[user].addr != None:
#if len(self.users[user].addr) != 2:
#return 0
if len(self.users[user].addr) == 2:
if self.users[user].addr is not None:
u_ip, u_port = self.users[user].addr
if u_ip != ip:
warning = _("IP %(ip)s:%(port)s is spoofing user %(user)s with a peer request, blocking because it does not match IP: %(real_ip)s") %{'ip':ip, 'port':port, 'user':user, 'real_ip':u_ip}
self.logMessage(warning , 1)
print warning
return 1
return 0
def ClosePeerConnection(self, peerconn, force=False):
if peerconn == None:
return
if not self.protothread.socketStillActive(peerconn.conn) or force:
self.queue.put(slskmessages.ConnClose(peerconn.conn))
self.del_peerconn(peerconn)
def UserInfoReply(self, msg):
try:
i = self.get_peerconn_by_conn(msg.conn.conn)
except KeyError:
print("Please help me: %s" % msg.conn.conn)
if self.userinfo is not None:
# probably impossible to do this
if i.username != self.config.sections["server"]["login"]:
self.userinfo.ShowInfo(i.username, msg)
def UserInfoRequest(self, msg):
user = None
ip = None
port = None
# Get peer's username, ip and port
try:
i = self.get_peerconn_by_conn(msg.conn.conn)
user = i.username
if i.addr is not None:
ip, port = i.addr
except KeyError:
print("Peers and apples: %s" % msg.conn.conn)
pass
if user == None:
# No peer connection
return
requestTime = time.time()
if user in self.requestedInfo:
if not requestTime > 10 + self.requestedInfo[user]:
# Ignoring request, because it's 10 or less seconds since the
# last one by this user
return
self.requestedInfo[user] = requestTime
# Check address is spoofed, if possible
#if self.CheckSpoof(user, ip, port):
# Message IS spoofed
# return
if user == self.config.sections["server"]["login"]:
if ip is not None and port is not None:
self.logMessage(_("Blocking %(user)s from making a UserInfo request, possible spoofing attempt from IP %(ip)s port %(port)s") %{'user':user, 'ip':ip, 'port':port}, 1)
else:
self.logMessage(_("Blocking %s from making a UserInfo request, possible spoofing attempt from an unknown IP & port") %(user), 1)
if msg.conn.conn != None:
self.queue.put(slskmessages.ConnClose(msg.conn.conn))
return
if user in self.config.sections["server"]["banlist"]:
self.logMessage(_("%(user)s is banned, but is making a UserInfo request") %{'user':user}, 1)
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 1)
return
try:
if sys.platform == "win32":
userpic = u"%s" % self.config.sections["userinfo"]["pic"]
if not os.path.exists(userpic):
userpic = self.config.sections["userinfo"]["pic"]
else:
userpic = self.config.sections["userinfo"]["pic"]
f=open(userpic,'rb')
pic = f.read()
f.close()
except:
pic = None
descr = self.encode(eval(self.config.sections["userinfo"]["descr"], {})).replace("\n", "\r\n")
if self.transfers is not None:
totalupl = self.transfers.getTotalUploadsAllowed()
queuesize = self.transfers.getUploadQueueSizes()[0]
slotsavail = self.transfers.allowNewUploads()
ua = self.frame.np.config.sections["transfers"]["remotedownloads"]
if ua:
uploadallowed = self.frame.np.config.sections["transfers"]["uploadallowed"]
else:
uploadallowed = ua
self.queue.put(slskmessages.UserInfoReply(msg.conn.conn, descr, pic, totalupl, queuesize, slotsavail, uploadallowed))
self.logMessage(_("%(user)s is making a UserInfo request") %{'user':user}, 1)
def SharedFileList(self, msg):
try:
i = self.get_peerconn_by_conn(msg.conn.conn)
except KeyError:
print("Kabloink:% s" % msg.conn.conn)
return
if self.userbrowse is not None:
if i.username != self.config.sections["server"]["login"]:
self.userbrowse.ShowInfo(i.username, msg)
def FileSearchResult(self, msg):
try:
i = self.get_peerconn_by_conn(msg.conn.conn)
except KeyError:
print("Booh, I'm lost! %s" % msg.conn.conn)
return
if self.search is not None:
if self.geoip:
if i.addr:
country = self.geoip.country_code_by_addr(i.addr[0])
else:
country = ""
else:
country = ""
if country is None:
country = ""
self.search.ShowResult(msg, i.username, country)
self.ClosePeerConnection(i.conn)
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def PierceFireWall(self, msg):
try:
i = self.peerconns_by_token[msg.token]
except KeyError:
print("I feel so depressed: %s" % msg.token)
return
if i.conn is None:
if i.conntimer is not None:
i.conntimer.cancel()
i.init.conn = msg.conn.conn
self.queue.put(i.init)
i.conn = msg.conn.conn
for j in i.msgs:
if j.__class__ is slskmessages.UserInfoRequest and self.userinfo is not None:
self.userinfo.InitWindow(i.username, msg.conn.conn)
if j.__class__ is slskmessages.GetSharedFileList and self.userbrowse is not None:
self.userbrowse.InitWindow(i.username, msg.conn.conn)
if j.__class__ is slskmessages.FileRequest and self.transfers is not None:
self.transfers.gotFileConnect(j.req, msg.conn.conn)
if j.__class__ is slskmessages.TransferRequest and self.transfers is not None:
self.transfers.gotConnect(j.req, msg.conn.conn)
j.conn = msg.conn.conn
self.queue.put(j)
i.msgs = []
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 3)
def CantConnectToPeer(self, msg):
try:
i = self.peerconns_by_token[msg.token]
except KeyError:
print("Look ma, no hands: %s" % msg.token)
return
if i.conntimer is not None:
i.conntimer.cancel()
if i == self.GetDistribConn():
self.DistribConnClosed(i)
self.del_peerconn(i)
self.logMessage(_("Can't connect to %s (either way), giving up") % (i.username), 3)
for j in i.msgs:
if j.__class__ in [slskmessages.TransferRequest, slskmessages.FileRequest] and self.transfers is not None:
self.transfers.gotCantConnect(j.req)
def ConnectToPeerTimeout(self, msg):
try:
i = self.get_peerconn_by_conn(msg.conn)
except:
print("Don't shoot the messenger")
return
if i == self.GetDistribConn():
self.DistribConnClosed(i)
self.del_peerconn(i)
self.logMessage(_("User %s does not respond to connect request, giving up") % (i.username), 3)
for j in i.msgs:
if j.__class__ in [slskmessages.TransferRequest, slskmessages.FileRequest] and self.transfers is not None:
self.transfers.gotCantConnect(j.req)
def TransferTimeout(self, msg):
if self.transfers is not None:
self.transfers.TransferTimeout(msg)
else:
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def FileDownload(self, msg):
if self.transfers is not None:
self.transfers.FileDownload(msg)
else:
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def FileUpload(self, msg):
if self.transfers is not None:
self.transfers.FileUpload(msg)
else:
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def FileRequest(self, msg):
if self.transfers is not None:
self.transfers.FileRequest(msg)
else:
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def FileError(self, msg):
if self.transfers is not None:
self.transfers.FileError(msg)
else:
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def TransferRequest(self, msg):
if self.transfers is not None:
self.transfers.TransferRequest(msg)
else:
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def TransferResponse(self, msg):
if self.transfers is not None:
self.transfers.TransferResponse(msg)
else:
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def QueueUpload(self, msg):
if self.transfers is not None:
self.transfers.QueueUpload(msg)
else:
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def QueueFailed(self, msg):
if self.transfers is not None:
self.transfers.QueueFailed(msg)
else:
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def PlaceholdUpload(self, msg):
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def PlaceInQueueRequest(self, msg):
if self.transfers is not None:
self.transfers.PlaceInQueueRequest(msg)
else:
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def UploadQueueNotification(self, msg):
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
self.transfers.UploadQueueNotification(msg)
def UploadFailed(self, msg):
if self.transfers is not None:
self.transfers.UploadFailed(msg)
else:
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def PlaceInQueue(self, msg):
if self.transfers is not None:
self.transfers.PlaceInQueue(msg)
else:
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def FolderContentsResponse(self, msg):
if self.transfers is not None:
try:
i = self.get_peerconn_by_conn(msg.conn.conn)
except KeyError:
print("Crash and burn: %s" % msg.conn.conn)
username = i.username
# Check for a large number of files
many = False
folder = ""
files = []
for i in msg.list.keys():
for j in msg.list[i].keys():
if os.path.commonprefix([i, j]) == j:
files = msg.list[i][j]
numfiles = len(files)
if numfiles > 100:
many = True
folder = j
if many:
self.frame.download_large_folder(username, folder, files, numfiles, msg)
else:
self.transfers.FolderContentsResponse(msg)
else:
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def RoomList(self, msg):
if self.chatrooms is not None:
self.chatrooms.roomsctrl.SetRoomList(msg)
self.setStatus("")
else:
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def GlobalUserList(self, msg):
if self.globallist is not None:
self.globallist.setGlobalUsersList(msg)
else:
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def PeerTransfer(self, msg):
if self.userinfo is not None and msg.msg is slskmessages.UserInfoReply:
self.userinfo.UpdateGauge(msg)
if self.userbrowse is not None and msg.msg is slskmessages.SharedFileList:
self.userbrowse.UpdateGauge(msg)
def TunneledMessage(self, msg):
if msg.code in self.protothread.peerclasses:
peermsg = self.protothread.peerclasses[msg.code](None)
peermsg.parseNetworkMessage(msg.msg)
peermsg.tunneleduser = msg.user
peermsg.tunneledreq = msg.req
peermsg.tunneledaddr = msg.addr
self.callback([peermsg])
else:
self.logMessage(_("Unknown tunneled message: %s") %(vars(msg)), 4)
def ExactFileSearch(self, msg):
''' Depreciated '''
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
try:
i = self.get_peerconn_by_conn(msg.conn.conn)
except KeyError:
print("...zen...: %s" % msg.conn.conn)
return
user = i.username
self.shares.processExactSearchRequest(msg.searchterm, user, msg.searchid, direct=1, checksum=msg.checksum)
def FileSearchRequest(self, msg):
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
try:
i = self.get_peerconns_by_conn(msg.conn.conn)
except KeyError:
print("Ploink: %s" % msg.conn.conn)
return
user = i.username
self.shares.processSearchRequest(msg.searchterm, user, msg.searchid, direct=1)
def SearchRequest(self, msg):
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
self.shares.processSearchRequest(msg.searchterm, msg.user, msg.searchid, direct=0)
def RoomSearchRequest(self, msg):
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
self.shares.processSearchRequest(msg.searchterm, msg.room, msg.searchid, direct=0)
def ToggleRespondDistributed(self, msg, settings=False):
"""
Toggle responding to distributed search each (default: 60sec)
interval
"""
if not self.config.sections["searches"]["search_results"]:
# Don't return _any_ results when this option is disabled
if self.respondDistributedTimer is not None:
self.respondDistributedTimer.cancel()
self.respondDistributed = False
return
if self.respondDistributedTimer is not None :
self.respondDistributedTimer.cancel()
if self.config.sections["searches"]["distrib_timer"]:
if not settings:
# Don't toggle when just changing the settings
self.respondDistributed = not self.respondDistributed
responddistributedtimeout = RespondToDistributedSearchesTimeout(self.callback)
self.respondDistributedTimer = threading.Timer(self.config.sections["searches"]["distrib_ignore"], responddistributedtimeout.timeout)
self.respondDistributedTimer.start()
else:
# Always respond
self.respondDistributed = True
def DistribSearch(self, msg):
if self.respondDistributed: # set in ToggleRespondDistributed
self.shares.processSearchRequest(msg.searchterm, msg.user, msg.searchid, 0)
def NetInfo(self, msg):
#print msg.list
self.distribcache.update(msg.list)
if len(self.distribcache) > 0:
self.queue.put(slskmessages.HaveNoParent(0))
if not self.GetDistribConn():
user = self.distribcache.keys()[0]
addr = self.distribcache[user]
self.queue.put(slskmessages.SearchParent( addr[0]))
self.ProcessRequestToPeer(user, slskmessages.DistribConn(), None, addr)
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def DistribAlive(self, msg):
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def GetDistribConn(self):
for i in self.peerconns:
if i.init.type == 'D':
return i
return None
def DistribConnClosed(self, conn):
del self.distribcache[conn.username]
if len(self.distribcache) > 0:
user = self.distribcache.keys()[0]
addr = self.distribcache[user]
self.queue.put(slskmessages.SearchParent( addr[0]))
#print user, addr
self.ProcessRequestToPeer(user, slskmessages.DistribConn(), None, addr)
else:
self.queue.put(slskmessages.HaveNoParent(1))
def GlobalRecommendations(self, msg):
self.frame.GlobalRecommendations(msg)
def Recommendations(self, msg):
self.frame.Recommendations(msg)
def ItemRecommendations(self, msg):
self.frame.ItemRecommendations(msg)
def SimilarUsers(self, msg):
self.frame.SimilarUsers(msg)
def ItemSimilarUsers(self, msg):
self.frame.ItemSimilarUsers(msg)
def RoomTickerState(self, msg):
if msg.room in self.config.sections["server"]["roomencoding"]:
encodings = [self.config.sections["server"]["roomencoding"][msg.room]] + self.config.sections["server"]["fallbackencodings"]
encodings.append(self.config.sections["server"]["enc"])
else:
encodings = [self.config.sections["server"]["enc"]] + self.config.sections["server"]["fallbackencodings"]
unicodes = {}
for user, bytes in msg.msgs.iteritems():
unicodes[user] = findBestEncoding(bytes, encodings)
msg.msgs = unicodes
if self.chatrooms is not None:
self.chatrooms.roomsctrl.TickerSet(msg)
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def RoomTickerAdd(self, msg):
if msg.room in self.config.sections["server"]["roomencoding"]:
encodings = [self.config.sections["server"]["roomencoding"][msg.room]] + self.config.sections["server"]["fallbackencodings"]
encodings.append(self.config.sections["server"]["enc"])
else:
encodings = [self.config.sections["server"]["enc"]] + self.config.sections["server"]["fallbackencodings"]
msg.msg = findBestEncoding(msg.msg, encodings)
if self.chatrooms is not None:
self.chatrooms.roomsctrl.TickerAdd(msg)
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def RoomTickerRemove(self, msg):
if self.chatrooms is not None:
self.chatrooms.roomsctrl.TickerRemove(msg)
self.logMessage("%s %s" %(msg.__class__, vars(msg)), 4)
def logTransfer(self, message, toUI = 0):
if self.config.sections["logging"]["transfers"]:
fn = os.path.join(self.config.sections["logging"]["logsdir"], "transfers.log")
try:
f = open(fn, "a")
f.write(time.strftime("%c"))
f.write(" %s\n" % message)
f.close()
except IOError, error:
self.logMessage(_("Couldn't write to transfer log: %s") % error)
if toUI:
self.logMessage(message)
class UserAddr:
def __init__(self, addr = None, behindfw = None, status = None):
self.addr = addr
self.behindfw = behindfw
self.status = status