2009-11-01 15:21:20 +00:00
|
|
|
# ##### BEGIN GPL LICENSE BLOCK #####
|
|
|
|
#
|
|
|
|
# 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 2
|
|
|
|
# of the License, or (at your option) any later version.
|
2009-12-29 00:04:57 +00:00
|
|
|
#
|
2009-11-01 15:21:20 +00:00
|
|
|
# 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.
|
2009-12-29 00:04:57 +00:00
|
|
|
#
|
2009-11-01 15:21:20 +00:00
|
|
|
# You should have received a copy of the GNU General Public License
|
|
|
|
# along with this program; if not, write to the Free Software Foundation,
|
2010-02-12 13:34:04 +00:00
|
|
|
# Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
2009-11-01 15:21:20 +00:00
|
|
|
#
|
|
|
|
# ##### END GPL LICENSE BLOCK #####
|
|
|
|
|
2010-01-06 18:56:24 +00:00
|
|
|
import sys, os, platform, shutil
|
2009-08-29 17:25:22 +00:00
|
|
|
import http, http.client, http.server, urllib
|
|
|
|
import subprocess, time
|
2010-09-12 14:04:54 +00:00
|
|
|
import json
|
2009-08-29 17:25:22 +00:00
|
|
|
|
2010-10-05 23:14:31 +00:00
|
|
|
import bpy
|
|
|
|
|
2009-08-29 17:25:22 +00:00
|
|
|
from netrender.utils import *
|
|
|
|
import netrender.model
|
2010-05-02 21:34:08 +00:00
|
|
|
import netrender.repath
|
2009-08-29 17:25:22 +00:00
|
|
|
|
2009-11-18 17:07:42 +00:00
|
|
|
BLENDER_PATH = sys.argv[0]
|
|
|
|
|
2009-08-29 17:25:22 +00:00
|
|
|
CANCEL_POLL_SPEED = 2
|
|
|
|
MAX_TIMEOUT = 10
|
|
|
|
INCREMENT_TIMEOUT = 1
|
2010-10-13 21:53:37 +00:00
|
|
|
try:
|
|
|
|
system = platform.system()
|
2010-10-13 22:20:34 +00:00
|
|
|
except UnicodeDecodeError:
|
2010-10-13 21:53:37 +00:00
|
|
|
import sys
|
|
|
|
system = sys.platform
|
2009-08-29 17:25:22 +00:00
|
|
|
|
2010-10-13 21:53:37 +00:00
|
|
|
if system in ('Windows', 'win32') and platform.version() >= '5': # Error mode is only available on Win2k or higher, that's version 5
|
2009-12-29 00:04:57 +00:00
|
|
|
import ctypes
|
|
|
|
def SetErrorMode():
|
|
|
|
val = ctypes.windll.kernel32.SetErrorMode(0x0002)
|
|
|
|
ctypes.windll.kernel32.SetErrorMode(val | 0x0002)
|
|
|
|
return val
|
|
|
|
|
|
|
|
def RestoreErrorMode(val):
|
|
|
|
ctypes.windll.kernel32.SetErrorMode(val)
|
2009-09-17 18:40:37 +00:00
|
|
|
else:
|
2009-12-29 00:04:57 +00:00
|
|
|
def SetErrorMode():
|
|
|
|
return 0
|
|
|
|
|
|
|
|
def RestoreErrorMode(val):
|
|
|
|
pass
|
2009-09-17 18:40:37 +00:00
|
|
|
|
2010-01-06 18:56:24 +00:00
|
|
|
def clearSlave(path):
|
|
|
|
shutil.rmtree(path)
|
|
|
|
|
2009-08-29 17:25:22 +00:00
|
|
|
def slave_Info():
|
2009-12-29 00:04:57 +00:00
|
|
|
sysname, nodename, release, version, machine, processor = platform.uname()
|
|
|
|
slave = netrender.model.RenderSlave()
|
|
|
|
slave.name = nodename
|
|
|
|
slave.stats = sysname + " " + release + " " + machine + " " + processor
|
|
|
|
return slave
|
2009-08-29 17:25:22 +00:00
|
|
|
|
2009-09-26 16:22:52 +00:00
|
|
|
def testCancel(conn, job_id, frame_number):
|
2009-12-29 00:04:57 +00:00
|
|
|
conn.request("HEAD", "/status", headers={"job-id":job_id, "job-frame": str(frame_number)})
|
2009-12-21 01:09:09 +00:00
|
|
|
|
2010-08-14 16:30:41 +00:00
|
|
|
# canceled if job isn't found anymore
|
|
|
|
if responseStatus(conn) == http.client.NO_CONTENT:
|
2009-12-29 00:04:57 +00:00
|
|
|
return True
|
|
|
|
else:
|
|
|
|
return False
|
2009-08-29 17:25:22 +00:00
|
|
|
|
2010-04-28 01:54:12 +00:00
|
|
|
def testFile(conn, job_id, slave_id, rfile, JOB_PREFIX, main_path = None):
|
|
|
|
job_full_path = prefixPath(JOB_PREFIX, rfile.filepath, main_path)
|
|
|
|
|
|
|
|
found = os.path.exists(job_full_path)
|
|
|
|
|
|
|
|
if found:
|
|
|
|
found_signature = hashFile(job_full_path)
|
|
|
|
found = found_signature == rfile.signature
|
|
|
|
|
|
|
|
if not found:
|
|
|
|
print("Found file %s at %s but signature mismatch!" % (rfile.filepath, job_full_path))
|
2010-05-02 21:34:08 +00:00
|
|
|
job_full_path = prefixPath(JOB_PREFIX, rfile.filepath, main_path, force = True)
|
2010-04-28 01:54:12 +00:00
|
|
|
|
|
|
|
if not found:
|
2010-07-17 18:40:00 +00:00
|
|
|
# Force prefix path if not found
|
|
|
|
job_full_path = prefixPath(JOB_PREFIX, rfile.filepath, main_path, force = True)
|
2010-08-14 16:30:41 +00:00
|
|
|
temp_path = os.path.join(JOB_PREFIX, "slave.temp")
|
2010-04-28 01:54:12 +00:00
|
|
|
conn.request("GET", fileURL(job_id, rfile.index), headers={"slave-id":slave_id})
|
2009-12-29 00:04:57 +00:00
|
|
|
response = conn.getresponse()
|
|
|
|
|
|
|
|
if response.status != http.client.OK:
|
|
|
|
return None # file for job not returned by server, need to return an error code to server
|
|
|
|
|
|
|
|
f = open(temp_path, "wb")
|
|
|
|
buf = response.read(1024)
|
|
|
|
|
|
|
|
while buf:
|
|
|
|
f.write(buf)
|
|
|
|
buf = response.read(1024)
|
|
|
|
|
|
|
|
f.close()
|
|
|
|
|
|
|
|
os.renames(temp_path, job_full_path)
|
2010-05-02 21:34:08 +00:00
|
|
|
|
|
|
|
rfile.filepath = job_full_path
|
2009-12-29 00:04:57 +00:00
|
|
|
|
|
|
|
return job_full_path
|
2009-09-01 01:09:05 +00:00
|
|
|
|
2010-01-08 17:52:21 +00:00
|
|
|
def render_slave(engine, netsettings, threads):
|
2009-12-29 00:04:57 +00:00
|
|
|
timeout = 1
|
|
|
|
|
|
|
|
engine.update_stats("", "Network render node initiation")
|
|
|
|
|
|
|
|
conn = clientConnection(netsettings.server_address, netsettings.server_port)
|
|
|
|
|
|
|
|
if conn:
|
2010-09-12 14:04:54 +00:00
|
|
|
conn.request("POST", "/slave", json.dumps(slave_Info().serialize()))
|
2009-12-29 00:04:57 +00:00
|
|
|
response = conn.getresponse()
|
2010-08-14 16:30:41 +00:00
|
|
|
response.read()
|
2009-12-29 00:04:57 +00:00
|
|
|
|
|
|
|
slave_id = response.getheader("slave-id")
|
|
|
|
|
2010-10-05 23:14:31 +00:00
|
|
|
NODE_PREFIX = os.path.join(bpy.path.abspath(netsettings.path), "slave_" + slave_id)
|
2009-12-29 00:04:57 +00:00
|
|
|
if not os.path.exists(NODE_PREFIX):
|
|
|
|
os.mkdir(NODE_PREFIX)
|
|
|
|
|
2010-04-27 03:14:40 +00:00
|
|
|
engine.update_stats("", "Network render connected to master, waiting for jobs")
|
|
|
|
|
2009-12-29 00:04:57 +00:00
|
|
|
while not engine.test_break():
|
|
|
|
conn.request("GET", "/job", headers={"slave-id":slave_id})
|
|
|
|
response = conn.getresponse()
|
|
|
|
|
|
|
|
if response.status == http.client.OK:
|
|
|
|
timeout = 1 # reset timeout on new job
|
|
|
|
|
2010-09-12 14:04:54 +00:00
|
|
|
job = netrender.model.RenderJob.materialize(json.loads(str(response.read(), encoding='utf8')))
|
2010-04-27 03:14:40 +00:00
|
|
|
engine.update_stats("", "Network render processing job from master")
|
2009-12-29 00:04:57 +00:00
|
|
|
|
2010-08-14 16:30:41 +00:00
|
|
|
JOB_PREFIX = os.path.join(NODE_PREFIX, "job_" + job.id)
|
2009-12-29 00:04:57 +00:00
|
|
|
if not os.path.exists(JOB_PREFIX):
|
|
|
|
os.mkdir(JOB_PREFIX)
|
|
|
|
|
|
|
|
|
|
|
|
if job.type == netrender.model.JOB_BLENDER:
|
|
|
|
job_path = job.files[0].filepath # path of main file
|
|
|
|
main_path, main_file = os.path.split(job_path)
|
|
|
|
|
2010-04-28 01:54:12 +00:00
|
|
|
job_full_path = testFile(conn, job.id, slave_id, job.files[0], JOB_PREFIX)
|
2009-12-29 00:04:57 +00:00
|
|
|
print("Fullpath", job_full_path)
|
|
|
|
print("File:", main_file, "and %i other files" % (len(job.files) - 1,))
|
|
|
|
|
|
|
|
for rfile in job.files[1:]:
|
2010-04-28 01:54:12 +00:00
|
|
|
testFile(conn, job.id, slave_id, rfile, JOB_PREFIX, main_path)
|
2010-05-02 21:34:08 +00:00
|
|
|
print("\t", rfile.filepath)
|
|
|
|
|
|
|
|
netrender.repath.update(job)
|
|
|
|
|
|
|
|
engine.update_stats("", "Render File "+ main_file+ " for job "+ job.id)
|
2009-12-29 00:04:57 +00:00
|
|
|
|
|
|
|
# announce log to master
|
|
|
|
logfile = netrender.model.LogFile(job.id, slave_id, [frame.number for frame in job.frames])
|
2010-09-12 14:04:54 +00:00
|
|
|
conn.request("POST", "/log", bytes(json.dumps(logfile.serialize()), encoding='utf8'))
|
2009-12-29 00:04:57 +00:00
|
|
|
response = conn.getresponse()
|
2010-08-14 16:30:41 +00:00
|
|
|
response.read()
|
2009-12-29 00:04:57 +00:00
|
|
|
|
|
|
|
|
|
|
|
first_frame = job.frames[0].number
|
|
|
|
|
|
|
|
# start render
|
|
|
|
start_t = time.time()
|
|
|
|
|
|
|
|
if job.type == netrender.model.JOB_BLENDER:
|
|
|
|
frame_args = []
|
|
|
|
|
|
|
|
for frame in job.frames:
|
|
|
|
print("frame", frame.number)
|
|
|
|
frame_args += ["-f", str(frame.number)]
|
|
|
|
|
|
|
|
val = SetErrorMode()
|
2010-08-14 16:30:41 +00:00
|
|
|
process = subprocess.Popen([BLENDER_PATH, "-b", "-noaudio", job_full_path, "-t", str(threads), "-o", os.path.join(JOB_PREFIX, "######"), "-E", "BLENDER_RENDER", "-F", "MULTILAYER"] + frame_args, stdout=subprocess.PIPE, stderr=subprocess.STDOUT)
|
2009-12-29 00:04:57 +00:00
|
|
|
RestoreErrorMode(val)
|
|
|
|
elif job.type == netrender.model.JOB_PROCESS:
|
|
|
|
command = job.frames[0].command
|
|
|
|
val = SetErrorMode()
|
|
|
|
process = subprocess.Popen(command.split(" "), stdout=subprocess.PIPE, stderr=subprocess.STDOUT)
|
|
|
|
RestoreErrorMode(val)
|
|
|
|
|
|
|
|
headers = {"slave-id":slave_id}
|
|
|
|
|
|
|
|
cancelled = False
|
|
|
|
stdout = bytes()
|
|
|
|
run_t = time.time()
|
2010-09-18 10:43:32 +00:00
|
|
|
while not cancelled and process.poll() is None:
|
2010-01-08 17:52:21 +00:00
|
|
|
stdout += process.stdout.read(1024)
|
2009-12-29 00:04:57 +00:00
|
|
|
current_t = time.time()
|
|
|
|
cancelled = engine.test_break()
|
|
|
|
if current_t - run_t > CANCEL_POLL_SPEED:
|
|
|
|
|
|
|
|
# update logs if needed
|
|
|
|
if stdout:
|
|
|
|
# (only need to update on one frame, they are linked
|
|
|
|
conn.request("PUT", logURL(job.id, first_frame), stdout, headers=headers)
|
|
|
|
response = conn.getresponse()
|
2010-08-14 16:30:41 +00:00
|
|
|
response.read()
|
2010-04-28 01:54:12 +00:00
|
|
|
|
|
|
|
# Also output on console
|
2010-08-20 06:09:58 +00:00
|
|
|
if netsettings.use_slave_output_log:
|
2010-04-28 01:54:12 +00:00
|
|
|
print(str(stdout, encoding='utf8'), end="")
|
2009-12-29 00:04:57 +00:00
|
|
|
|
|
|
|
stdout = bytes()
|
|
|
|
|
|
|
|
run_t = current_t
|
|
|
|
if testCancel(conn, job.id, first_frame):
|
|
|
|
cancelled = True
|
|
|
|
|
2010-05-02 21:34:08 +00:00
|
|
|
if job.type == netrender.model.JOB_BLENDER:
|
|
|
|
netrender.repath.reset(job)
|
|
|
|
|
2009-12-29 00:04:57 +00:00
|
|
|
# read leftovers if needed
|
|
|
|
stdout += process.stdout.read()
|
|
|
|
|
|
|
|
if cancelled:
|
|
|
|
# kill process if needed
|
2010-09-18 10:43:32 +00:00
|
|
|
if process.poll() is None:
|
2010-08-21 17:07:19 +00:00
|
|
|
try:
|
|
|
|
process.terminate()
|
|
|
|
except OSError:
|
|
|
|
pass
|
2009-12-29 00:04:57 +00:00
|
|
|
continue # to next frame
|
|
|
|
|
2010-04-28 01:54:12 +00:00
|
|
|
# flush the rest of the logs
|
|
|
|
if stdout:
|
|
|
|
# Also output on console
|
2010-08-20 06:09:58 +00:00
|
|
|
if netsettings.use_slave_thumb:
|
2010-04-28 01:54:12 +00:00
|
|
|
print(str(stdout, encoding='utf8'), end="")
|
|
|
|
|
|
|
|
# (only need to update on one frame, they are linked
|
|
|
|
conn.request("PUT", logURL(job.id, first_frame), stdout, headers=headers)
|
2010-08-14 16:30:41 +00:00
|
|
|
if responseStatus(conn) == http.client.NO_CONTENT:
|
2010-04-28 01:54:12 +00:00
|
|
|
continue
|
|
|
|
|
2009-12-29 00:04:57 +00:00
|
|
|
total_t = time.time() - start_t
|
|
|
|
|
|
|
|
avg_t = total_t / len(job.frames)
|
|
|
|
|
|
|
|
status = process.returncode
|
|
|
|
|
|
|
|
print("status", status)
|
|
|
|
|
|
|
|
headers = {"job-id":job.id, "slave-id":slave_id, "job-time":str(avg_t)}
|
|
|
|
|
|
|
|
|
|
|
|
if status == 0: # non zero status is error
|
|
|
|
headers["job-result"] = str(DONE)
|
|
|
|
for frame in job.frames:
|
|
|
|
headers["job-frame"] = str(frame.number)
|
|
|
|
if job.type == netrender.model.JOB_BLENDER:
|
|
|
|
# send image back to server
|
2010-01-09 18:21:27 +00:00
|
|
|
|
2010-08-14 16:30:41 +00:00
|
|
|
filename = os.path.join(JOB_PREFIX, "%06d.exr" % frame.number)
|
2010-01-31 14:46:28 +00:00
|
|
|
|
2010-01-09 18:21:27 +00:00
|
|
|
# thumbnail first
|
2010-08-20 06:09:58 +00:00
|
|
|
if netsettings.use_slave_thumb:
|
2010-01-09 18:21:27 +00:00
|
|
|
thumbname = thumbnail(filename)
|
2010-01-31 14:46:28 +00:00
|
|
|
|
2010-01-09 18:21:27 +00:00
|
|
|
f = open(thumbname, 'rb')
|
|
|
|
conn.request("PUT", "/thumb", f, headers=headers)
|
|
|
|
f.close()
|
2010-08-14 16:30:41 +00:00
|
|
|
responseStatus(conn)
|
|
|
|
|
2010-01-09 18:21:27 +00:00
|
|
|
|
|
|
|
f = open(filename, 'rb')
|
2009-12-29 00:04:57 +00:00
|
|
|
conn.request("PUT", "/render", f, headers=headers)
|
|
|
|
f.close()
|
2010-08-14 16:30:41 +00:00
|
|
|
if responseStatus(conn) == http.client.NO_CONTENT:
|
2009-12-29 00:04:57 +00:00
|
|
|
continue
|
2010-01-31 14:46:28 +00:00
|
|
|
|
2009-12-29 00:04:57 +00:00
|
|
|
elif job.type == netrender.model.JOB_PROCESS:
|
|
|
|
conn.request("PUT", "/render", headers=headers)
|
2010-08-14 16:30:41 +00:00
|
|
|
if responseStatus(conn) == http.client.NO_CONTENT:
|
2009-12-29 00:04:57 +00:00
|
|
|
continue
|
|
|
|
else:
|
|
|
|
headers["job-result"] = str(ERROR)
|
|
|
|
for frame in job.frames:
|
|
|
|
headers["job-frame"] = str(frame.number)
|
|
|
|
# send error result back to server
|
|
|
|
conn.request("PUT", "/render", headers=headers)
|
2010-08-14 16:30:41 +00:00
|
|
|
if responseStatus(conn) == http.client.NO_CONTENT:
|
2009-12-29 00:04:57 +00:00
|
|
|
continue
|
2010-04-27 03:14:40 +00:00
|
|
|
|
|
|
|
engine.update_stats("", "Network render connected to master, waiting for jobs")
|
2009-12-29 00:04:57 +00:00
|
|
|
else:
|
|
|
|
if timeout < MAX_TIMEOUT:
|
|
|
|
timeout += INCREMENT_TIMEOUT
|
|
|
|
|
|
|
|
for i in range(timeout):
|
|
|
|
time.sleep(1)
|
|
|
|
if engine.test_break():
|
2010-01-06 18:56:24 +00:00
|
|
|
break
|
2009-12-29 00:04:57 +00:00
|
|
|
|
|
|
|
conn.close()
|
2010-01-31 14:46:28 +00:00
|
|
|
|
2010-08-20 06:09:58 +00:00
|
|
|
if netsettings.use_slave_clear:
|
2010-01-06 18:56:24 +00:00
|
|
|
clearSlave(NODE_PREFIX)
|
2009-11-18 17:07:42 +00:00
|
|
|
|
|
|
|
if __name__ == "__main__":
|
2009-12-29 00:04:57 +00:00
|
|
|
pass
|