fix to install
This commit is contained in:
commit
3dcb2cfd52
5 changed files with 193 additions and 99 deletions
|
|
@ -25,7 +25,8 @@ class DocType:
|
|||
if not validate_email_add(self.doc.email):
|
||||
msgprint("%s is not a valid email id" % self.doc.email)
|
||||
raise Exception
|
||||
self.doc.name = self.doc.email
|
||||
|
||||
self.doc.name = self.doc.email
|
||||
|
||||
def on_update(self):
|
||||
# owner is always name
|
||||
|
|
|
|||
|
|
@ -21,6 +21,10 @@ def copy_defs():
|
|||
#
|
||||
class Installer:
|
||||
def __init__(self, root_login, root_password):
|
||||
|
||||
import webnotes
|
||||
import webnotes.db
|
||||
import webnotes.defs
|
||||
|
||||
self.root_password = root_password
|
||||
from webnotes.model.db_schema import DbManager
|
||||
|
|
@ -29,12 +33,14 @@ class Installer:
|
|||
webnotes.conn=self.conn
|
||||
webnotes.session= {'user':'Administrator'}
|
||||
self.dbman = DbManager(self.conn)
|
||||
self.mysql_path = hasattr(defs, 'mysql_path') and webnotes.defs.mysql_path or ''
|
||||
self.mysql_path = hasattr(webnotes.defs, 'mysql_path') and webnotes.defs.mysql_path or ''
|
||||
|
||||
#
|
||||
# run framework related cleanups
|
||||
#
|
||||
def framework_cleanups(self, target):
|
||||
|
||||
import webnotes
|
||||
self.dbman.drop_table('__DocTypeCache')
|
||||
webnotes.conn.sql("create table `__DocTypeCache` (name VARCHAR(120), modified DATETIME, content TEXT, server_code_compiled TEXT)")
|
||||
|
||||
|
|
@ -48,6 +54,7 @@ class Installer:
|
|||
Imports the "Core" module from .txt file and creates
|
||||
Creates profile Administrator
|
||||
"""
|
||||
import webnotes
|
||||
from webnotes.modules.import_module import import_module
|
||||
from webnotes.modules.module_manager import reload_doc
|
||||
|
||||
|
|
@ -99,12 +106,13 @@ class Installer:
|
|||
"""
|
||||
a very simplified version, just for the time being..will eventually be deprecated once the framework stabilizes.
|
||||
"""
|
||||
import webnotes.defs
|
||||
|
||||
# delete user (if exists)
|
||||
self.dbman.delete_user(target)
|
||||
|
||||
# create user and db
|
||||
self.dbman.create_user(target,getattr(defs,'db_password',None))
|
||||
self.dbman.create_user(target,getattr(webnotes.defs,'db_password',None))
|
||||
if verbose: print "Created user %s" % target
|
||||
|
||||
# create a database
|
||||
|
|
@ -158,7 +166,7 @@ def make_scheduler(root_login, root_password, verbose):
|
|||
dbman.delete_user('master_scheduler')
|
||||
|
||||
# create user and db
|
||||
dbman.create_user('master_scheduler', getattr(defs,'db_password',None))
|
||||
dbman.create_user('master_scheduler', getattr(webnotes.defs,'db_password',None))
|
||||
if verbose: print "Created user master_scheduler"
|
||||
|
||||
# create a database
|
||||
|
|
@ -214,14 +222,14 @@ if __name__=='__main__':
|
|||
|
||||
try:
|
||||
|
||||
from webnotes import defs
|
||||
import webnotes
|
||||
import webnotes.db
|
||||
import webnotes.defs
|
||||
except ImportError:
|
||||
copy_defs()
|
||||
from webnotes import defs
|
||||
import webnotes
|
||||
import webnotes.db
|
||||
import webnotes.defs
|
||||
|
||||
if len(args)==3:
|
||||
|
||||
|
|
|
|||
2
cgi-bin/webnotes/model/doc.py
Normal file → Executable file
2
cgi-bin/webnotes/model/doc.py
Normal file → Executable file
|
|
@ -227,7 +227,7 @@ class Document:
|
|||
|
||||
self.name = self.name.strip() # no leading and trailing blanks
|
||||
|
||||
forbidden = ['%', "'", '"', ',', '#', '*', '?', '&', '`']
|
||||
forbidden = ['%', "'", '"', '#', '*', '?', '`']
|
||||
for f in forbidden:
|
||||
if f in self.name:
|
||||
webnotes.msgprint('%s not allowed in ID (name)' % f, raise_exception =1)
|
||||
|
|
|
|||
|
|
@ -1,107 +1,190 @@
|
|||
import webnotes
|
||||
"""
|
||||
This module handles the On Demand Backup utility
|
||||
"""
|
||||
#Imports
|
||||
import os, webnotes
|
||||
from datetime import datetime
|
||||
|
||||
backup_folder = '/backups'
|
||||
mysql_path = ''
|
||||
download_folder = 'backups'
|
||||
|
||||
def mysqldump(db, folder=''):
|
||||
global mysql_path
|
||||
import os
|
||||
import webnotes.defs
|
||||
|
||||
os.system('%(path)smysqldump %(db)s > %(folder)s%(db)s.sql -u %(db)s -p%(pwd)s --ignore-table=%(db)s.__DocTypeCache' % {'path':mysql_path, 'db':db, 'pwd':webnotes.defs.db_password, 'folder':folder})
|
||||
#Global constants
|
||||
from webnotes.defs import backup_path, backup_link_path, backup_url
|
||||
verbose = 0
|
||||
|
||||
def backup_db(db, conn, from_all=0):
|
||||
import os
|
||||
global backup_folder
|
||||
|
||||
try:
|
||||
# Check processlist
|
||||
if from_all or len(conn.sql("show processlist")) == 1:
|
||||
p = backup_folder
|
||||
if from_all: p = backup_folder + '/dumps'
|
||||
|
||||
# clear old file
|
||||
os.system('rm %s/%s.tar.gz' % (p,db))
|
||||
|
||||
# dump
|
||||
mysqldump(db, p+'/')
|
||||
|
||||
# zip
|
||||
os.system('tar czf %s/%s.tar.gz %s/%s.sql' % (p, db, p, db))
|
||||
os.system('rm %s/%s.sql' % (p, db))
|
||||
else:
|
||||
print("Another process is running in database. Please try again")
|
||||
except Exception, e:
|
||||
#sql('unlock tables')
|
||||
raise e
|
||||
#-------------------------------------------------------------------------------
|
||||
class BackupGenerator:
|
||||
"""
|
||||
This class contains methods to perform On Demand Backup
|
||||
|
||||
def backup_all():
|
||||
# backups folder
|
||||
import os
|
||||
import webnotes.db
|
||||
global backup_folder
|
||||
|
||||
conn = webnotes.db.Database(use_default=1)
|
||||
dblist = conn.sql('select db_name from tabAccount')
|
||||
To initialize, specify (db_name, user, password, db_file_name=None)
|
||||
If specifying db_file_name, also append ".sql.gz"
|
||||
"""
|
||||
def __init__(self, db_name, user, password, db_file_name=None):
|
||||
self.db_name = db_name
|
||||
self.user = user
|
||||
self.password = password
|
||||
self.db_file_name = db_file_name and db_file_name \
|
||||
or (backup_path + db_name + ".sql.gz")
|
||||
|
||||
# backup -all in /backups folder
|
||||
for d in (('accounts',),) + dblist:
|
||||
backup_db(d[0], conn, 1)
|
||||
def take_dump(self):
|
||||
"""
|
||||
Dumps a db via mysqldump
|
||||
"""
|
||||
os.system("""mysqldump -u %(user)s -p%(password)s %(db_name)s |
|
||||
gzip -c > %(db_file_name)s""" % self.__dict__)
|
||||
|
||||
conn.close()
|
||||
|
||||
# dump all in /daily folder
|
||||
import time, datetime
|
||||
fname = 'daily-' + time.strftime('%Y-%m-%d') + '.tar.gz'
|
||||
def copy_to_backup_link(self):
|
||||
"""
|
||||
Copies the backup file from backup path to shared link path
|
||||
It also gives the file a random name, thus hiding the db_name
|
||||
"""
|
||||
import random
|
||||
todays_date = "".join(str(datetime.date(datetime.today())).split("-"))
|
||||
random_number = str(int(random.random()*99999999))
|
||||
|
||||
#Generate a random name using today's date and a 8 digit random number
|
||||
random_name = todays_date + "_" + random_number + ".sql.gz"
|
||||
|
||||
os.system("""cp -f %(src_file)s %(dest_file)s""" % \
|
||||
{"src_file":self.db_file_name,
|
||||
"dest_file":(backup_link_path + random_name)})
|
||||
if verbose: print "file copied"
|
||||
return random_name
|
||||
|
||||
# daily dump
|
||||
os.system('tar czf %s/daily/%s %s/dumps' % (backup_folder, fname, backup_folder))
|
||||
|
||||
# keep only three files
|
||||
if len(os.listdir(backup_folder + '/daily')) > 3:
|
||||
delete_oldest_file(backup_folder + '/daily')
|
||||
|
||||
# if sunday, then copy to weekly
|
||||
if datetime.datetime.now().weekday()==6:
|
||||
os.system('cp '+backup_folder+'/daily/'+fname+' '+backup_folder+'/weekly/'+fname)
|
||||
|
||||
# keep only three files
|
||||
if len(os.listdir(backup_folder + '/weekly')) > 3:
|
||||
delete_oldest_file(backup_folder + '/weekly')
|
||||
|
||||
def delete_oldest_file(folder):
|
||||
import os
|
||||
a = sorted(os.listdir(folder), key=lambda fn: os.stat(folder+'/'+fn).st_mtime, reverse=False)
|
||||
if a:
|
||||
os.system('rm %s/%s' % (folder, a[0]))
|
||||
|
||||
def get_recipients(self):
|
||||
"""
|
||||
Get recepient's email address
|
||||
"""
|
||||
#import webnotes.db
|
||||
#webnotes.conn = webnotes.db.Database(use_default=1)
|
||||
recipient_list = webnotes.conn.sql(\
|
||||
"""SELECT parent FROM tabUserRole
|
||||
WHERE role='System Manager'
|
||||
AND parent!='Administrator'
|
||||
AND parent IN
|
||||
(SELECT email FROM tabProfile WHERE enabled=1)""")
|
||||
return [i[0] for i in recipient_list]
|
||||
|
||||
|
||||
def send_email(self, backup_file):
|
||||
"""
|
||||
Sends the link to backup file located at erpnext/backups
|
||||
"""
|
||||
file_url = backup_url + backup_file
|
||||
from webnotes.utils.email_lib import sendmail
|
||||
|
||||
recipient_list = self.get_recipients()
|
||||
msg = """<a href=%(file_url)s>Click here to begin downloading\
|
||||
your backup</a>
|
||||
|
||||
This link will be valid for 24 hours.
|
||||
|
||||
Also, a new backup will be available for download (if requested)\
|
||||
only after 24 hours.""" % {"file_url":file_url}
|
||||
|
||||
datetime_str = datetime.fromtimestamp(os.stat(self.db_file_name).st_ctime)
|
||||
|
||||
subject = datetime_str.strftime("%d/%m/%Y %H:%M:%S") + """ - Backup ready to be downloaded"""
|
||||
sendmail(recipients=recipient_list, msg=msg, subject=subject)
|
||||
return recipient_list
|
||||
|
||||
|
||||
def get_backup(self):
|
||||
"""
|
||||
Takes a new dump if existing file is old
|
||||
and sends the link to the file as email
|
||||
"""
|
||||
#Check if file exists and is less than a day old
|
||||
#If not Take Dump
|
||||
if is_file_old(self.db_file_name):
|
||||
self.take_dump()
|
||||
|
||||
#Copy file to backup_link_path
|
||||
#This is a security hole. When the user calls get_backup repeatedly
|
||||
#a file will be generated each time.
|
||||
backup_file = self.copy_to_backup_link()
|
||||
|
||||
#Email Link
|
||||
recipient_list = self.send_email(backup_file)
|
||||
|
||||
return recipient_list
|
||||
|
||||
#-------------------------------------------------------------------------------
|
||||
def get_backup():
|
||||
import webnotes
|
||||
import os, time
|
||||
"""
|
||||
This function is executed when the user clicks on
|
||||
Toos > Download Backup
|
||||
"""
|
||||
#if verbose: print webnotes.conn.cur_db_name + " " + webnotes.defs.db_password
|
||||
odb = BackupGenerator(webnotes.conn.cur_db_name, webnotes.conn.cur_db_name,\
|
||||
webnotes.defs.db_password)
|
||||
recipient_list = odb.get_backup()
|
||||
delete_temp_backups()
|
||||
webnotes.msgprint("""A download link to your backup will be emailed \
|
||||
to you shortly on the following email address:
|
||||
%s""" % (str(recipient_list),))
|
||||
|
||||
global backup_folder, download_folder
|
||||
|
||||
# get the last nightly backup file from the backups folder
|
||||
os.chdir(download_folder)
|
||||
def delete_temp_backups():
|
||||
"""
|
||||
Cleans up the backup_link_path directory by deleting files older than 24 hours
|
||||
"""
|
||||
file_list = os.listdir(backup_link_path)
|
||||
for this_file in file_list:
|
||||
this_file_path = backup_link_path + this_file
|
||||
if is_file_old(this_file_path):
|
||||
os.remove(this_file_path)
|
||||
|
||||
if webnotes.conn.cur_db_name:
|
||||
fname = webnotes.conn.cur_db_name + '.tar.gz'
|
||||
|
||||
# rename it
|
||||
from random import choice
|
||||
lnd='0123456789'
|
||||
new_name = ''.join(map(lambda x,y=lnd: choice(y), range(8))) + '.sql.gz'
|
||||
folder = backup_folder + '/dumps/'
|
||||
def is_file_old(db_file_name, older_than=24):
|
||||
"""
|
||||
Checks if file exists and is older than specified hours
|
||||
Returns ->
|
||||
True: file does not exist or file is old
|
||||
False: file is new
|
||||
"""
|
||||
if os.path.isfile(db_file_name):
|
||||
from datetime import timedelta
|
||||
import time
|
||||
#Get timestamp of the file
|
||||
file_datetime = datetime.fromtimestamp\
|
||||
(os.stat(db_file_name).st_ctime)
|
||||
if datetime.today() - file_datetime >= timedelta(hours = older_than):
|
||||
if verbose: print "File is old"
|
||||
return True
|
||||
else:
|
||||
if verbose: print "File is recent"
|
||||
return False
|
||||
else:
|
||||
if verbose: print "File does not exist"
|
||||
return True
|
||||
|
||||
# get the newest file
|
||||
if os.path.exists(folder):
|
||||
os.system('cp '+ folder + webnotes.conn.cur_db_name+'.sql.gz' + ' ./' + new_name)
|
||||
webnotes.msgprint('Your nightly backup is available for download by <a href="'+download_folder+'/' + new_name + '">clicking here</a> (only for the next few hours)')
|
||||
#-------------------------------------------------------------------------------
|
||||
|
||||
if __name__ == "__main__":
|
||||
"""
|
||||
is_file_old db_name user password
|
||||
get_backup db_name user password
|
||||
"""
|
||||
import sys
|
||||
cmd = sys.argv[1]
|
||||
if cmd == "is_file_old":
|
||||
odb = BackupGenerator(sys.argv[2], sys.argv[3], sys.argv[4])
|
||||
is_file_old(odb.db_file_name)
|
||||
|
||||
# delete any files older than a day
|
||||
now = time.time()
|
||||
for f in os.listdir('.'):
|
||||
if os.stat(f).st_mtime < (now - 86400):
|
||||
if os.path.isfile(f):
|
||||
os.remove(f)
|
||||
if cmd == "get_backup":
|
||||
odb = BackupGenerator(sys.argv[2], sys.argv[3], sys.argv[4])
|
||||
odb.get_backup()
|
||||
|
||||
if cmd == "take_dump":
|
||||
odb = BackupGenerator(sys.argv[2], sys.argv[3], sys.argv[4])
|
||||
odb.take_dump()
|
||||
|
||||
if cmd == "send_email":
|
||||
odb = BackupGenerator(sys.argv[2], sys.argv[3], sys.argv[4])
|
||||
odb.send_email("abc.sql.gz")
|
||||
|
||||
if cmd == "delete_temp_backups":
|
||||
delete_temp_backups()
|
||||
|
||||
|
|
|
|||
|
|
@ -148,9 +148,11 @@ class FormEmail:
|
|||
self.email.add_attachment(self.dn.replace(' ','').replace('/','-') + '.html', self.body)
|
||||
|
||||
# attachments
|
||||
# self.with_attachments comes from http form variables
|
||||
# i.e. with_attachments=1
|
||||
if cint(self.with_attachments):
|
||||
for a in self.set_attachments():
|
||||
a and self.email.attach(a.split(',')[0])
|
||||
a and self.email.attach_file(a.split(',')[0])
|
||||
|
||||
# cc
|
||||
if self.cc:
|
||||
|
|
|
|||
Loading…
Add table
Reference in a new issue