| 3 | 4 |
new file mode 100644 |
| ... | ... |
@@ -0,0 +1,50 @@ |
| 1 |
+"use strict"; |
|
| 2 |
+ |
|
| 3 |
+const type = "image/jpeg"; |
|
| 4 |
+const quality = 0.95; |
|
| 5 |
+const width = 1920; |
|
| 6 |
+const height = 1440; |
|
| 7 |
+ |
|
| 8 |
+const camera = document.getElementById("camera");
|
|
| 9 |
+const message = document.createElement("div");
|
|
| 10 |
+const video = document.createElement("video");
|
|
| 11 |
+const canvas = document.createElement("canvas")
|
|
| 12 |
+const ctx = canvas.getContext("2d");
|
|
| 13 |
+ |
|
| 14 |
+let facingMode = "environment"; |
|
| 15 |
+ |
|
| 16 |
+async function getVideo() {
|
|
| 17 |
+ try {
|
|
| 18 |
+ video.srcObject = await navigator.mediaDevices.getUserMedia({
|
|
| 19 |
+ video: { facingMode, width, height },
|
|
| 20 |
+ }); |
|
| 21 |
+ video.play(); |
|
| 22 |
+ } catch (err) {
|
|
| 23 |
+ camera.removeChild(video); |
|
| 24 |
+ message.innerHTML = `Could not get video: ${err.name}: ${err.message}`;
|
|
| 25 |
+ } |
|
| 26 |
+} |
|
| 27 |
+ |
|
| 28 |
+async function post() |
|
| 29 |
+{
|
|
| 30 |
+ try {
|
|
| 31 |
+ message.innerHTML = "Posting..."; |
|
| 32 |
+ canvas.width = video.videoWidth; |
|
| 33 |
+ canvas.height = video.videoHeight; |
|
| 34 |
+ ctx.drawImage(video, 0, 0); |
|
| 35 |
+ const blob = await new Promise((r) => canvas.toBlob(r, type, quality)); |
|
| 36 |
+ const resp = await fetch("/", {
|
|
| 37 |
+ method: "POST", |
|
| 38 |
+ body: blob, |
|
| 39 |
+ }); |
|
| 40 |
+ message.innerHTML = `${resp.status}: ${resp.statusText}`;
|
|
| 41 |
+ } catch (err) {
|
|
| 42 |
+ message.innerHTML = `${err.name}: ${err.message}`;
|
|
| 43 |
+ } |
|
| 44 |
+} |
|
| 45 |
+ |
|
| 46 |
+message.innerHTML = "Press video to post."; |
|
| 47 |
+video.addEventListener("click", post);
|
|
| 48 |
+camera.appendChild(message); |
|
| 49 |
+camera.appendChild(video); |
|
| 50 |
+getVideo(); |
| ... | ... |
@@ -1,6 +1,8 @@ |
| 1 | 1 |
#!/usr/bin/env python3 |
| 2 | 2 |
|
| 3 | 3 |
import sys |
| 4 |
+import os |
|
| 5 |
+import datetime |
|
| 4 | 6 |
import http.server |
| 5 | 7 |
import ssl |
| 6 | 8 |
|
| ... | ... |
@@ -8,13 +10,26 @@ import ssl |
| 8 | 10 |
PORT = 8000 |
| 9 | 11 |
HOST = "" |
| 10 | 12 |
PUBLIC = "public" |
| 13 |
+CAMERA = "camera" |
|
| 11 | 14 |
|
| 12 | 15 |
|
| 13 | 16 |
class Handler(http.server.SimpleHTTPRequestHandler): |
| 14 | 17 |
|
| 15 | 18 |
def __init__(self, *args, **kwargs): |
| 19 |
+ os.makedirs(CAMERA, exist_ok=True) |
|
| 16 | 20 |
super().__init__(*args, **kwargs, directory=PUBLIC) |
| 17 | 21 |
|
| 22 |
+ def do_POST(self): |
|
| 23 |
+ length = int(self.headers["Content-Length"]) |
|
| 24 |
+ ext = str(self.headers["Content-Type"]).split("/")[1]
|
|
| 25 |
+ base = datetime.datetime.now().strftime("%Y-%m-%d_%H-%M-%S_%f")
|
|
| 26 |
+ name = f"{CAMERA}/{base}.{ext}"
|
|
| 27 |
+ self.log_message("Receiving '%s'", name)
|
|
| 28 |
+ with open(name, "wb") as f: |
|
| 29 |
+ f.write(self.rfile.read(length)) |
|
| 30 |
+ self.send_response(200, name) |
|
| 31 |
+ self.end_headers() |
|
| 32 |
+ |
|
| 18 | 33 |
|
| 19 | 34 |
args = iter(sys.argv[1:]) |
| 20 | 35 |
port = int(next(args, PORT)) |