-
-
Save humanium/fa65f0f4340faab1a2f5 to your computer and use it in GitHub Desktop.
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
#!/usr/bin/env python2 | |
# coding: utf-8 | |
import os,socket,threading,time | |
import traceback | |
currdir=os.path.abspath('.') | |
class FTPserverThread(threading.Thread): | |
def __init__(self,(conn,addr)): | |
self.conn=conn | |
self.addr=addr | |
self.basewd=currdir | |
self.cwd=self.basewd | |
self.rest=False | |
threading.Thread.__init__(self) | |
def run(self): | |
self.conn.send('220 Welcome!\r\n') | |
while True: | |
cmd=self.conn.recv(256) | |
if not cmd: break | |
else: | |
print 'Recieved:',cmd | |
try: | |
func=getattr(self,cmd[:4].strip().upper()) | |
func(cmd) | |
except Exception,e: | |
print 'ERROR:',e | |
#traceback.print_exc() | |
self.conn.send('500 Sorry.\r\n') | |
def USER(self,cmd): | |
self.conn.send('331 OK.\r\n') | |
def PASS(self,cmd): | |
self.conn.send('230 OK.\r\n') | |
def QUIT(self,cmd): | |
self.conn.send('221 Goodbye.\r\n') | |
def NOOP(self,cmd): | |
self.conn.send('200 OK.\r\n') | |
def TYPE(self,cmd): | |
self.mode=cmd[5] | |
self.conn.send('200 Binary mode.\r\n') | |
def CDUP(self,cmd): | |
if not os.path.samefile(self.cwd,self.basewd): | |
#learn from stackoverflow | |
self.cwd=os.path.abspath(os.path.join(self.cwd,'..')) | |
self.conn.send('200 OK.\r\n') | |
def PWD(self,cmd): | |
cwd=os.path.relpath(self.cwd,self.basewd) | |
if cwd=='.': | |
cwd='/' | |
else: | |
cwd='/'+cwd | |
self.conn.send('257 \"%s\"\r\n' % cwd) | |
def CWD(self,cmd): | |
chwd=cmd[4:-2] | |
if chwd=='/': | |
self.cwd=self.basewd | |
elif chwd[0]=='/': | |
self.cwd=os.path.join(self.basewd,chwd[1:]) | |
else: | |
self.cwd=os.path.join(self.cwd,chwd) | |
self.conn.send('250 OK.\r\n') | |
def PORT(self,cmd): | |
l=cmd[5:].split(',') | |
self.dataAddr='.'.join(l[:4]) | |
self.dataPort=(int(l[4])<<8)+int(l[5]) | |
self.conn.send('200 Get port.\r\n') | |
def LIST(self,cmd): | |
self.conn.send('150 Here comes the directory listing.\r\n') | |
self.datasock=socket.socket(socket.AF_INET,socket.SOCK_STREAM) | |
self.datasock.connect((self.dataAddr,self.dataPort)) | |
print self.cwd | |
for t in os.listdir(self.cwd): | |
k=self.toListItem(os.path.join(self.cwd,t)) | |
self.datasock.send(k+'\r\n') | |
self.datasock.close() | |
self.conn.send('226 Directory send OK.\r\n') | |
def toListItem(self,fn): | |
st=os.stat(fn) | |
fullmode='rwxrwxrwx' | |
mode='' | |
for i in range(9): | |
mode+=((st.st_mode>>(8-i))&1) and fullmode[i] or '-' | |
d=(os.path.isdir(fn)) and 'd' or '-' | |
ftime=time.strftime(' %b %d %H:%M ', time.gmtime(st.st_mtime)) | |
return d+mode+' 1 user group '+str(st.st_size)+ftime+os.path.basename(fn) | |
def MKD(self,cmd): | |
dn=os.path.join(self.cwd,cmd[4:-2]) | |
os.mkdir(dn) | |
self.conn.send('257 Directory created.\r\n') | |
def RMD(self,cmd): | |
dn=os.path.join(self.cwd,cmd[4:-2]) | |
#os.rmdir(dn) | |
self.conn.send('250 Directory deleted.\r\n') | |
def DELE(self,cmd): | |
fn=os.path.join(self.cwd,cmd[5:-2]) | |
#os.remove(fn) | |
self.conn.send('250 File deleted.\r\n') | |
def RNFR(self,cmd): | |
self.rnfn=os.path.join(self.cwd,cmd[5:-2]) | |
self.conn.send('350 Ready.\r\n') | |
def RNTO(self,cmd): | |
fn=os.path.join(self.cwd,cmd[5:-2]) | |
os.rename(self.rnfn,fn) | |
self.conn.send('250 File renamed.\r\n') | |
def REST(self,cmd): | |
self.pos=int(cmd[5:-2]) | |
self.rest=True | |
self.conn.send('250 File position reseted.\r\n') | |
def RETR(self,cmd): | |
fn=os.path.join(self.cwd,cmd[5:-2]) | |
print 'Downlowding:',fn | |
if self.mode=='I': | |
fi=open(fn,'rb') | |
else: | |
fi=open(fn,'r') | |
self.conn.send('150 Opening data connection.\r\n') | |
self.datasock=socket.socket(socket.AF_INET,socket.SOCK_STREAM) | |
self.datasock.connect((self.dataAddr,self.dataPort)) | |
if self.rest: | |
fi.seek(self.pos) | |
self.rest=False | |
data= fi.read(1024) | |
while data: | |
self.datasock.send(data) | |
data=fi.read(1024) | |
fi.close() | |
self.datasock.close() | |
self.conn.send('226 Transfer complete.\r\n') | |
def STOR(self,cmd): | |
fn=os.path.join(self.cwd,cmd[5:-2]) | |
print 'Uplaoding:',fn | |
if self.mode=='I': | |
fo=open(fn,'wb') | |
else: | |
fo=open(fn,'w') | |
self.conn.send('150 Opening data connection.\r\n') | |
self.datasock=socket.socket(socket.AF_INET,socket.SOCK_STREAM) | |
self.datasock.connect((self.dataAddr,self.dataPort)) | |
while True: | |
data=self.datasock.recv(1024) | |
if not data: break | |
fo.write(data) | |
fo.close() | |
self.datasock.close() | |
self.conn.send('226 Transfer complete.\r\n') | |
class FTPserver(threading.Thread): | |
def __init__(self): | |
self.sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM) | |
self.sock.bind(('',8888)) | |
threading.Thread.__init__(self) | |
def run(self): | |
self.sock.listen(5) | |
while True: | |
th=FTPserverThread(self.sock.accept()) | |
th.daemon=True | |
th.start() | |
def stop(self): | |
self.sock.close() | |
if __name__=='__main__': | |
ftp=FTPserver() | |
ftp.daemon=True | |
ftp.start() | |
raw_input('Enter to end...\n') | |
ftp.stop() |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment