* persistent notification on ongoing timelapse render job (#485) * non-colliding timelapse snapshot name generation to not delete existing snapshots when new print starts and timelapse has not yet been rendered, also only delete snapshots if timelapse rendered (#318) * list of unrendered timelapses, with option to delete files or to render timelapse
154 lines
4.6 KiB
Python
154 lines
4.6 KiB
Python
# coding=utf-8
|
|
from __future__ import absolute_import
|
|
|
|
__author__ = "Gina Häußge <osd@foosel.net>"
|
|
__license__ = 'GNU Affero General Public License http://www.gnu.org/licenses/agpl.html'
|
|
__copyright__ = "Copyright (C) 2014 The OctoPrint Project - Released under terms of the AGPLv3 License"
|
|
|
|
import os
|
|
|
|
from flask import request, jsonify, url_for, make_response
|
|
from werkzeug.utils import secure_filename
|
|
|
|
import octoprint.timelapse
|
|
import octoprint.util as util
|
|
from octoprint.settings import settings, valid_boolean_trues
|
|
|
|
from octoprint.server import admin_permission, printer
|
|
from octoprint.server.util.flask import redirect_to_tornado, restricted_access, get_json_command_from_request
|
|
from octoprint.server.api import api
|
|
|
|
from octoprint.server import NO_CONTENT
|
|
|
|
|
|
#~~ timelapse handling
|
|
|
|
|
|
@api.route("/timelapse", methods=["GET"])
|
|
def getTimelapseData():
|
|
timelapse = octoprint.timelapse.current
|
|
|
|
config = {"type": "off"}
|
|
if timelapse is not None and isinstance(timelapse, octoprint.timelapse.ZTimelapse):
|
|
config["type"] = "zchange"
|
|
config["postRoll"] = timelapse.post_roll
|
|
config["fps"] = timelapse.fps
|
|
elif timelapse is not None and isinstance(timelapse, octoprint.timelapse.TimedTimelapse):
|
|
config["type"] = "timed"
|
|
config["postRoll"] = timelapse.post_roll
|
|
config["fps"] = timelapse.fps
|
|
config.update({
|
|
"interval": timelapse.interval
|
|
})
|
|
|
|
files = octoprint.timelapse.get_finished_timelapses()
|
|
for file in files:
|
|
file["url"] = url_for("index") + "downloads/timelapse/" + file["name"]
|
|
|
|
result = dict(config=config,
|
|
files=files)
|
|
|
|
if "unrendered" in request.values and request.values["unrendered"] in valid_boolean_trues:
|
|
result.update(unrendered=octoprint.timelapse.get_unrendered_timelapses())
|
|
|
|
return jsonify(result)
|
|
|
|
|
|
@api.route("/timelapse/<filename>", methods=["GET"])
|
|
def downloadTimelapse(filename):
|
|
return redirect_to_tornado(request, url_for("index") + "downloads/timelapse/" + filename)
|
|
|
|
|
|
@api.route("/timelapse/<filename>", methods=["DELETE"])
|
|
@restricted_access
|
|
def deleteTimelapse(filename):
|
|
if util.is_allowed_file(filename, ["mpg"]):
|
|
timelapse_folder = settings().getBaseFolder("timelapse")
|
|
full_path = os.path.realpath(os.path.join(timelapse_folder, filename))
|
|
if full_path.startswith(timelapse_folder) and os.path.exists(full_path):
|
|
os.remove(full_path)
|
|
return getTimelapseData()
|
|
|
|
|
|
@api.route("/timelapse/unrendered/<name>", methods=["DELETE"])
|
|
@restricted_access
|
|
def deleteUnrenderedTimelapse(name):
|
|
octoprint.timelapse.delete_unrendered_timelapse(name)
|
|
return NO_CONTENT
|
|
|
|
|
|
@api.route("/timelapse/unrendered/<name>", methods=["POST"])
|
|
@restricted_access
|
|
def processUnrenderedTimelapseCommand(name):
|
|
# valid file commands, dict mapping command name to mandatory parameters
|
|
valid_commands = {
|
|
"render": []
|
|
}
|
|
|
|
command, data, response = get_json_command_from_request(request, valid_commands)
|
|
if response is not None:
|
|
return response
|
|
|
|
if command == "render":
|
|
if printer.is_printing() or printer.is_paused():
|
|
return make_response("Printer is currently printing, cannot render timelapse", 409)
|
|
octoprint.timelapse.render_unrendered_timelapse(name)
|
|
|
|
return NO_CONTENT
|
|
|
|
|
|
@api.route("/timelapse", methods=["POST"])
|
|
@restricted_access
|
|
def setTimelapseConfig():
|
|
if "type" in request.values:
|
|
config = {
|
|
"type": request.values["type"],
|
|
"postRoll": 0,
|
|
"fps": 25,
|
|
"options": {}
|
|
}
|
|
|
|
if "postRoll" in request.values:
|
|
try:
|
|
postRoll = int(request.values["postRoll"])
|
|
except ValueError:
|
|
return make_response("Invalid value for postRoll: %r" % request.values["postRoll"], 400)
|
|
else:
|
|
if postRoll >= 0:
|
|
config["postRoll"] = postRoll
|
|
else:
|
|
return make_response("Invalid value for postRoll: %d" % postRoll, 400)
|
|
|
|
if "fps" in request.values:
|
|
try:
|
|
fps = int(request.values["fps"])
|
|
except ValueError:
|
|
return make_response("Invalid value for fps: %r" % request.values["fps"], 400)
|
|
else:
|
|
if fps > 0:
|
|
config["fps"] = fps
|
|
else:
|
|
return make_response("Invalid value for fps: %d" % fps, 400)
|
|
|
|
if "interval" in request.values:
|
|
config["options"] = {
|
|
"interval": 10
|
|
}
|
|
|
|
try:
|
|
interval = int(request.values["interval"])
|
|
except ValueError:
|
|
return make_response("Invalid value for interval: %r" % request.values["interval"])
|
|
else:
|
|
if interval > 0:
|
|
config["options"]["interval"] = interval
|
|
else:
|
|
return make_response("Invalid value for interval: %d" % interval)
|
|
|
|
if admin_permission.can() and "save" in request.values and request.values["save"] in valid_boolean_trues:
|
|
octoprint.timelapse.configure_timelapse(config, True)
|
|
else:
|
|
octoprint.timelapse.configure_timelapse(config)
|
|
|
|
return getTimelapseData()
|
|
|