Exchange ids vs posts. Support image paste.
This commit is contained in:
89
db.js
89
db.js
@@ -118,6 +118,50 @@ export async function addDataArray(userID, array) {
|
|||||||
console.error('Error in opening database:', error);
|
console.error('Error in opening database:', error);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
export async function checkPostIds(userID, post_ids) {
|
||||||
|
try {
|
||||||
|
const db = await openDatabase(userID);
|
||||||
|
const transaction = db.transaction(postStoreName, "readwrite");
|
||||||
|
const store = transaction.objectStore(postStoreName);
|
||||||
|
const index = store.index("postIDIndex");
|
||||||
|
transaction.oncomplete = () => {
|
||||||
|
console.log("Transaction completed successfully");
|
||||||
|
db.close();
|
||||||
|
};
|
||||||
|
transaction.onerror = (event) => {
|
||||||
|
console.error("Transaction error:", event.target.error);
|
||||||
|
db.close();
|
||||||
|
};
|
||||||
|
let postIdsNeeded = [];
|
||||||
|
for (let id of post_ids) {
|
||||||
|
try {
|
||||||
|
let havePost = await new Promise((resolve, reject) => {
|
||||||
|
const getRequest = index.getKey(id);
|
||||||
|
getRequest.onerror = (e) => {
|
||||||
|
console.log(e.target.error);
|
||||||
|
reject(e);
|
||||||
|
};
|
||||||
|
getRequest.onsuccess = async (e) => {
|
||||||
|
const key = e.target.result;
|
||||||
|
resolve(key !== undefined);
|
||||||
|
};
|
||||||
|
});
|
||||||
|
// console.log(post.post_id, havePost);
|
||||||
|
if (!havePost) {
|
||||||
|
postIdsNeeded.push(id);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (error) {
|
||||||
|
console.error("Error processing post:", error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
console.log(`checkPostIds need ${postIdsNeeded.length} posts`);
|
||||||
|
return postIdsNeeded;
|
||||||
|
}
|
||||||
|
catch (error) {
|
||||||
|
console.error("Error in opening database:", error);
|
||||||
|
}
|
||||||
|
}
|
||||||
export async function mergeDataArray(userID, array) {
|
export async function mergeDataArray(userID, array) {
|
||||||
try {
|
try {
|
||||||
const db = await openDatabase(userID);
|
const db = await openDatabase(userID);
|
||||||
@@ -215,4 +259,49 @@ export async function getAllData(userID) {
|
|||||||
};
|
};
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
export async function getAllIds(userID) {
|
||||||
|
const db = await openDatabase(userID);
|
||||||
|
const transaction = db.transaction(postStoreName, "readonly");
|
||||||
|
const store = transaction.objectStore(postStoreName);
|
||||||
|
const index = store.index("postIDIndex");
|
||||||
|
let keys = [];
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
let request = index.openKeyCursor();
|
||||||
|
request.onsuccess = (event) => {
|
||||||
|
let cursor = event.target.result;
|
||||||
|
if (cursor) {
|
||||||
|
keys.push(cursor.key);
|
||||||
|
cursor.continue();
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
resolve(keys);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
request.onerror = (event) => {
|
||||||
|
reject(event);
|
||||||
|
};
|
||||||
|
});
|
||||||
|
}
|
||||||
|
export async function getPostsByIds(userID, postIDs) {
|
||||||
|
const db = await openDatabase(userID);
|
||||||
|
const transaction = db.transaction(postStoreName, "readonly");
|
||||||
|
const store = transaction.objectStore(postStoreName);
|
||||||
|
const index = store.index("postIDIndex");
|
||||||
|
let posts = [];
|
||||||
|
for (const postID of postIDs) {
|
||||||
|
const post = await new Promise((resolve, reject) => {
|
||||||
|
let request = index.get(postID);
|
||||||
|
request.onsuccess = (event) => {
|
||||||
|
resolve(event.target.result); // Resolve with the post
|
||||||
|
};
|
||||||
|
request.onerror = (event) => {
|
||||||
|
reject(event); // Reject if any error occurs
|
||||||
|
};
|
||||||
|
});
|
||||||
|
if (post) {
|
||||||
|
posts.push(post); // Add the post to the result array if found
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return posts; // Return the array of posts
|
||||||
|
}
|
||||||
//# sourceMappingURL=db.js.map
|
//# sourceMappingURL=db.js.map
|
||||||
@@ -23,11 +23,11 @@
|
|||||||
|
|
||||||
<div class="flex-container">
|
<div class="flex-container">
|
||||||
<div class="content">
|
<div class="content">
|
||||||
<div class="img-button" id="ddln-logo-button"><img class="logo" src="favicon.ico"></div>
|
<div class="img-button" id="ddln_logo_button"><img class="logo" src="favicon.ico"></div>
|
||||||
|
|
||||||
<div id="status"></div>
|
<div id="status"></div>
|
||||||
|
|
||||||
<div id="info" style="display:block">
|
<div id="info" style="display:none">
|
||||||
<div id="profile">
|
<div id="profile">
|
||||||
<span class="form_label">username:</span><span class="form_field" id="username" contenteditable="true">unnamed</span>
|
<span class="form_label">username:</span><span class="form_field" id="username" contenteditable="true">unnamed</span>
|
||||||
</div>
|
</div>
|
||||||
@@ -51,6 +51,9 @@
|
|||||||
|
|
||||||
<textarea cols="60" rows="6" id="textarea_post"></textarea>
|
<textarea cols="60" rows="6" id="textarea_post"></textarea>
|
||||||
<div class="button">
|
<div class="button">
|
||||||
|
<input type="file" id="fileInput" multiple style="display:none">
|
||||||
|
|
||||||
|
<button id="button_add_pic" >🏞️</button>
|
||||||
<button id="button_post" >post</button>
|
<button id="button_post" >post</button>
|
||||||
</div>
|
</div>
|
||||||
<!-- <div id="torrent-content"></div> -->
|
<!-- <div id="torrent-content"></div> -->
|
||||||
|
|||||||
142
main.css
Normal file
142
main.css
Normal file
@@ -0,0 +1,142 @@
|
|||||||
|
body {
|
||||||
|
font-family: sans-serif;
|
||||||
|
color: rgb(202, 208, 211);
|
||||||
|
background-color: black;
|
||||||
|
/* Use the font with a fallback */
|
||||||
|
}
|
||||||
|
|
||||||
|
hr {
|
||||||
|
border-color: rgb(60, 60, 60);
|
||||||
|
}
|
||||||
|
|
||||||
|
.form_field {
|
||||||
|
font-size: medium;
|
||||||
|
font-family: sans-serif;
|
||||||
|
background-color: rgb(0, 0, 0);
|
||||||
|
color: rgb(202, 208, 211);
|
||||||
|
width: 100%;
|
||||||
|
box-sizing: border-box;
|
||||||
|
padding-left: 5px;
|
||||||
|
padding-right: 5px;
|
||||||
|
border: 1px solid rgb(132, 136, 138);
|
||||||
|
resize: vertical;
|
||||||
|
border-radius: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#textarea_post {
|
||||||
|
font-size: medium;
|
||||||
|
font-family: sans-serif;
|
||||||
|
background-color: rgb(0, 0, 0);
|
||||||
|
color: rgb(202, 208, 211);
|
||||||
|
width: 100%;
|
||||||
|
box-sizing: border-box;
|
||||||
|
padding-left: 30px;
|
||||||
|
padding-right: 30px;
|
||||||
|
padding-top: 10px;
|
||||||
|
border: 1px solid rgb(132, 136, 138);
|
||||||
|
resize: vertical;
|
||||||
|
border-radius: 40px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.flex-container {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: flex-start;
|
||||||
|
min-height: 100vh;
|
||||||
|
padding: 0px;
|
||||||
|
/* Add some padding around the flex container */
|
||||||
|
}
|
||||||
|
|
||||||
|
.content {
|
||||||
|
max-width: 600px;
|
||||||
|
/* Your preferred max width for the content */
|
||||||
|
flex: 1;
|
||||||
|
/* Shorthand for flex-grow, flex-shrink and flex-basis */
|
||||||
|
min-width: 300px;
|
||||||
|
/* Minimum width the content can shrink to */
|
||||||
|
padding: 20px;
|
||||||
|
box-shadow: 0 0 5px rgb(60, 60, 60);
|
||||||
|
text-align: left;
|
||||||
|
overflow-x: hidden;
|
||||||
|
/* Hide horizontal overflow inside the flex container */
|
||||||
|
line-height: 1.3;
|
||||||
|
}
|
||||||
|
|
||||||
|
.embed {
|
||||||
|
max-width: 800px;
|
||||||
|
border-color: red;
|
||||||
|
border: 1px, solid;
|
||||||
|
padding: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.postImage {
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
#log {
|
||||||
|
font-family: monospace;
|
||||||
|
text-wrap: nowrap;
|
||||||
|
font-size: 10px;
|
||||||
|
margin-bottom: 20px;
|
||||||
|
height: 150px;
|
||||||
|
width: 50%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.button {
|
||||||
|
text-align: right;
|
||||||
|
}
|
||||||
|
|
||||||
|
#buttons {
|
||||||
|
margin-left: 40px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#button_post {
|
||||||
|
margin-right: 40px;
|
||||||
|
}
|
||||||
|
|
||||||
|
a {
|
||||||
|
color: rgb(29, 155, 240);
|
||||||
|
}
|
||||||
|
|
||||||
|
.logo {
|
||||||
|
width: 32px;
|
||||||
|
height: 32px;
|
||||||
|
image-rendering: pixelated;
|
||||||
|
}
|
||||||
|
|
||||||
|
#torrent-content {
|
||||||
|
border: solid 1px;
|
||||||
|
width: 800px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.img-button {
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
button {
|
||||||
|
background-color: rgb(0, 0, 0);
|
||||||
|
border-radius: 10px;
|
||||||
|
padding-left: 10px;
|
||||||
|
padding-right: 10px;
|
||||||
|
padding-top: 5px;
|
||||||
|
padding-bottom: 5px;
|
||||||
|
margin-left: 5px;
|
||||||
|
color: rgb(255, 255, 255);
|
||||||
|
/* border:solid 1px white; */
|
||||||
|
border: 1px solid rgb(132, 136, 138);
|
||||||
|
color: rgb(202, 208, 211);
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
video {
|
||||||
|
width:100%
|
||||||
|
}
|
||||||
|
|
||||||
|
iframe {
|
||||||
|
width: 100%
|
||||||
|
}
|
||||||
|
|
||||||
|
.qrcode_image {
|
||||||
|
background-color: white;
|
||||||
|
padding: 10px;
|
||||||
|
}
|
||||||
82
main.go
82
main.go
@@ -69,21 +69,21 @@ type Peer struct {
|
|||||||
lastActive time.Time
|
lastActive time.Time
|
||||||
}
|
}
|
||||||
|
|
||||||
func removePeer(peerID string, peer *Peer) {
|
// func removePeer(peerID string, peer *Peer) {
|
||||||
delete(peerConnections, peerID)
|
// delete(peerConnections, peerID)
|
||||||
|
|
||||||
for userID, peers := range userPeers {
|
// for userID, peers := range userPeers {
|
||||||
delete(peers, peerID)
|
// delete(peers, peerID)
|
||||||
if len(peers) == 0 {
|
// if len(peers) == 0 {
|
||||||
delete(userPeers, userID)
|
// delete(userPeers, userID)
|
||||||
}
|
// }
|
||||||
}
|
// }
|
||||||
|
|
||||||
delete(connectionPeers, peer.conn)
|
// delete(connectionPeers, peer.conn)
|
||||||
|
|
||||||
// Close the peer's send channel
|
// // Close the peer's send channel
|
||||||
close(peer.send)
|
// close(peer.send)
|
||||||
}
|
// }
|
||||||
|
|
||||||
func handleWebSocket(w http.ResponseWriter, r *http.Request) {
|
func handleWebSocket(w http.ResponseWriter, r *http.Request) {
|
||||||
log.Println("Websocket connection!", r.RemoteAddr)
|
log.Println("Websocket connection!", r.RemoteAddr)
|
||||||
@@ -348,39 +348,39 @@ func main() {
|
|||||||
Handler: nil, // Use the default ServeMux
|
Handler: nil, // Use the default ServeMux
|
||||||
}
|
}
|
||||||
|
|
||||||
// Start the inactivity monitor goroutine
|
// // Start the inactivity monitor goroutine
|
||||||
go func() {
|
// go func() {
|
||||||
ticker := time.NewTicker(10 * time.Second)
|
// ticker := time.NewTicker(10 * time.Second)
|
||||||
defer ticker.Stop()
|
// defer ticker.Stop()
|
||||||
|
|
||||||
for {
|
// for {
|
||||||
select {
|
// select {
|
||||||
case <-done:
|
// case <-done:
|
||||||
return
|
// return
|
||||||
case <-ticker.C:
|
// case <-ticker.C:
|
||||||
now := time.Now()
|
// now := time.Now()
|
||||||
|
|
||||||
// Collect inactive peers
|
// // Collect inactive peers
|
||||||
var inactivePeers []string
|
// var inactivePeers []string
|
||||||
for peerID, peer := range peerConnections {
|
// for peerID, peer := range peerConnections {
|
||||||
if now.Sub(peer.lastActive) > 60*time.Second {
|
// if now.Sub(peer.lastActive) > 60*time.Second {
|
||||||
inactivePeers = append(inactivePeers, peerID)
|
// inactivePeers = append(inactivePeers, peerID)
|
||||||
}
|
// }
|
||||||
}
|
// }
|
||||||
|
|
||||||
// Remove inactive peers
|
// // Remove inactive peers
|
||||||
for _, peerID := range inactivePeers {
|
// for _, peerID := range inactivePeers {
|
||||||
peer := peerConnections[peerID]
|
// peer := peerConnections[peerID]
|
||||||
|
|
||||||
if peer != nil {
|
// if peer != nil {
|
||||||
log.Printf("Peer %s inactive for more than 60 seconds. Closing connection.", peerID)
|
// log.Printf("Peer %s inactive for more than 60 seconds. Closing connection.", peerID)
|
||||||
peer.conn.Close()
|
// peer.conn.Close()
|
||||||
removePeer(peerID, peer)
|
// removePeer(peerID, peer)
|
||||||
}
|
// }
|
||||||
}
|
// }
|
||||||
}
|
// }
|
||||||
}
|
// }
|
||||||
}()
|
// }()
|
||||||
|
|
||||||
// Run a goroutine to handle graceful shutdown
|
// Run a goroutine to handle graceful shutdown
|
||||||
go func() {
|
go func() {
|
||||||
|
|||||||
201
main.js
201
main.js
@@ -1,5 +1,5 @@
|
|||||||
// TODO: virtual list, only rerender what's needed so things can keep playing.
|
// TODO: virtual list, only rerender what's needed so things can keep playing.
|
||||||
import { getData, addData, addDataArray, clearData, deleteData, mergeDataArray, getAllData } from "./db.js";
|
import { getData, addData, addDataArray, clearData, deleteData, mergeDataArray, checkPostIds, getAllIds, getPostsByIds } from "./db.js";
|
||||||
// let posts:any;
|
// let posts:any;
|
||||||
// let keyBase = "dandelion_posts_v1_"
|
// let keyBase = "dandelion_posts_v1_"
|
||||||
// let key:string = "";
|
// let key:string = "";
|
||||||
@@ -58,7 +58,6 @@ window.addEventListener('scroll', () => {
|
|||||||
if (scrollPoint >= totalPageHeight) {
|
if (scrollPoint >= totalPageHeight) {
|
||||||
console.log('Scrolled to the bottom!');
|
console.log('Scrolled to the bottom!');
|
||||||
console.log(scrollPoint, totalPageHeight);
|
console.log(scrollPoint, totalPageHeight);
|
||||||
// You can perform your action here
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
// let peer = await new PeerConnection(peer_id);
|
// let peer = await new PeerConnection(peer_id);
|
||||||
@@ -73,6 +72,23 @@ window.addEventListener('scroll', () => {
|
|||||||
// this.addPosts(newPosts);
|
// this.addPosts(newPosts);
|
||||||
// }
|
// }
|
||||||
// }
|
// }
|
||||||
|
function arrayBufferToBase64(buffer) {
|
||||||
|
var binary = '';
|
||||||
|
var bytes = new Uint8Array(buffer);
|
||||||
|
var len = bytes.byteLength;
|
||||||
|
for (var i = 0; i < len; i++) {
|
||||||
|
binary += String.fromCharCode(bytes[i]);
|
||||||
|
}
|
||||||
|
return window.btoa(binary);
|
||||||
|
}
|
||||||
|
function base64ToArrayBuffer(base64) {
|
||||||
|
var binaryString = atob(base64);
|
||||||
|
var bytes = new Uint8Array(binaryString.length);
|
||||||
|
for (var i = 0; i < binaryString.length; i++) {
|
||||||
|
bytes[i] = binaryString.charCodeAt(i);
|
||||||
|
}
|
||||||
|
return bytes.buffer;
|
||||||
|
}
|
||||||
class wsConnection {
|
class wsConnection {
|
||||||
send(message) {
|
send(message) {
|
||||||
let json = "";
|
let json = "";
|
||||||
@@ -82,7 +98,7 @@ class wsConnection {
|
|||||||
catch (e) {
|
catch (e) {
|
||||||
console.log(e, "wsConnection send: Couldn't serialize message", message);
|
console.log(e, "wsConnection send: Couldn't serialize message", message);
|
||||||
}
|
}
|
||||||
log(`ws->${json.slice(0, 80)}`);
|
log(`ws->${json.slice(0, 240)}`);
|
||||||
this.websocket.send(json);
|
this.websocket.send(json);
|
||||||
}
|
}
|
||||||
helloResponseHandler(data) {
|
helloResponseHandler(data) {
|
||||||
@@ -96,19 +112,58 @@ class wsConnection {
|
|||||||
type: "peer_message",
|
type: "peer_message",
|
||||||
from: this.peerID,
|
from: this.peerID,
|
||||||
to: peerID,
|
to: peerID,
|
||||||
message: { type: "get_posts_for_user", user_id: userID }
|
message: { type: "get_post_ids_for_user", user_id: userID }
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
pongHandler(data) {
|
pongHandler(data) {
|
||||||
}
|
}
|
||||||
async getPostsForUserResponseHandler(data) {
|
async getPostIdsForUserResponseHandler(data) {
|
||||||
// log(`getPostsForUserResponse: ${data}`)
|
// log(`getPostsForUserResponse: ${data}`)
|
||||||
|
let message = data.message;
|
||||||
|
console.log(`getPostIdsForUserResponseHandler Got ${message.post_ids.length} from peer ${data.from}`);
|
||||||
|
console.log(`Checking post IDs...`);
|
||||||
|
let postIds = await checkPostIds(message.user_id, data.message.post_ids);
|
||||||
|
if (postIds.length === 0) {
|
||||||
|
log(`Don't need any posts from peer ${data.from}`);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
let responseMessage = { type: "peer_message", from: app.peerID, to: data.from, message: { type: "get_posts_for_user", post_ids: postIds, user_id: message.user_id } };
|
||||||
|
this.send(responseMessage);
|
||||||
|
}
|
||||||
|
async getPostIdsForUserHandler(data) {
|
||||||
|
let message = data.message;
|
||||||
|
let postIds = await getAllIds(message.user_id) ?? [];
|
||||||
|
let responseMessage = { type: "peer_message", from: app.peerID, to: data.from, message: { type: "get_post_ids_for_user_response", post_ids: postIds, user_id: message.user_id } };
|
||||||
|
this.send(responseMessage);
|
||||||
|
}
|
||||||
|
// Send posts to peer
|
||||||
|
async getPostsForUserHandler(data) {
|
||||||
|
let message = data.message;
|
||||||
|
let posts = await getPostsByIds(message.user_id, message.post_ids) ?? [];
|
||||||
|
let output = [];
|
||||||
|
for (let post of posts) {
|
||||||
|
let newPost = post.data;
|
||||||
|
if (newPost.image_data) {
|
||||||
|
newPost.image_data = arrayBufferToBase64(newPost.image_data);
|
||||||
|
}
|
||||||
|
output.push(newPost);
|
||||||
|
}
|
||||||
|
// posts = posts.map((post:any)=>{let newPost = post.data; if (newPost.image_data){newPost.image_data = arraybufferto};return newPost});
|
||||||
|
// posts = posts.map((post:any)=>{})
|
||||||
|
let responseMessage = { type: "peer_message", from: app.peerID, to: data.from, message: { type: "get_posts_for_user_response", posts: output, user_id: message.user_id } };
|
||||||
|
this.send(responseMessage);
|
||||||
|
}
|
||||||
|
// Got posts from peer
|
||||||
|
async getPostsForUserReponseHandler(data) {
|
||||||
let message = data.message;
|
let message = data.message;
|
||||||
console.log(`getPostsForUserResponseHandler Got ${message.posts.length} from peer ${data.from}`);
|
console.log(`getPostsForUserResponseHandler Got ${message.posts.length} from peer ${data.from}`);
|
||||||
for (let post of message.posts) {
|
for (let post of message.posts) {
|
||||||
post.post_timestamp = new Date(post.post_timestamp);
|
post.post_timestamp = new Date(post.post_timestamp);
|
||||||
|
if (post.image_data) {
|
||||||
|
post.image_data = base64ToArrayBuffer(post.image_data);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
console.log(`Merging same user peer posts...`);
|
console.log(`Merging same user peer posts...`);
|
||||||
await mergeDataArray(message.user_id, data.message.posts);
|
await mergeDataArray(message.user_id, data.message.posts);
|
||||||
@@ -117,14 +172,6 @@ class wsConnection {
|
|||||||
app.render(app.posts);
|
app.render(app.posts);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
async getPostsForUserHandler(data) {
|
|
||||||
let message = data.message;
|
|
||||||
let posts = await getAllData(message.user_id) ?? []; // this doesn't get all posts!
|
|
||||||
posts = posts.map((post) => post.data);
|
|
||||||
// let posts = await getAllData(message.user_id) ?? [];
|
|
||||||
let responseMessage = { type: "peer_message", from: app.peerID, to: data.from, message: { type: "get_posts_for_user_response", posts: posts, user_id: message.user_id } };
|
|
||||||
this.send(responseMessage);
|
|
||||||
}
|
|
||||||
async peerMessageHandler(data) {
|
async peerMessageHandler(data) {
|
||||||
log(`peerMessageHandler ${data}`);
|
log(`peerMessageHandler ${data}`);
|
||||||
let peerMessageType = data.message.type;
|
let peerMessageType = data.message.type;
|
||||||
@@ -168,7 +215,7 @@ class wsConnection {
|
|||||||
window.setTimeout(() => { this.connect(); }, this.retry * 1000);
|
window.setTimeout(() => { this.connect(); }, this.retry * 1000);
|
||||||
};
|
};
|
||||||
this.websocket.onmessage = (event) => {
|
this.websocket.onmessage = (event) => {
|
||||||
log('ws:<-' + event.data.slice(0, 80));
|
log('ws:<-' + event.data.slice(0, 240));
|
||||||
let data = JSON.parse(event.data);
|
let data = JSON.parse(event.data);
|
||||||
let { type } = data;
|
let { type } = data;
|
||||||
let handler = this.messageHandlers.get(type);
|
let handler = this.messageHandlers.get(type);
|
||||||
@@ -200,8 +247,10 @@ class wsConnection {
|
|||||||
this.messageHandlers.set('hello', this.helloResponseHandler.bind(this));
|
this.messageHandlers.set('hello', this.helloResponseHandler.bind(this));
|
||||||
this.messageHandlers.set('pong', this.pongHandler);
|
this.messageHandlers.set('pong', this.pongHandler);
|
||||||
this.messageHandlers.set('peer_message', this.peerMessageHandler.bind(this));
|
this.messageHandlers.set('peer_message', this.peerMessageHandler.bind(this));
|
||||||
|
this.peerMessageHandlers.set('get_post_ids_for_user', this.getPostIdsForUserHandler.bind(this));
|
||||||
|
this.peerMessageHandlers.set('get_post_ids_for_user_response', this.getPostIdsForUserResponseHandler.bind(this));
|
||||||
this.peerMessageHandlers.set('get_posts_for_user', this.getPostsForUserHandler.bind(this));
|
this.peerMessageHandlers.set('get_posts_for_user', this.getPostsForUserHandler.bind(this));
|
||||||
this.peerMessageHandlers.set('get_posts_for_user_response', this.getPostsForUserResponseHandler.bind(this));
|
this.peerMessageHandlers.set('get_posts_for_user_response', this.getPostsForUserReponseHandler.bind(this));
|
||||||
this.connect();
|
this.connect();
|
||||||
if (!this.websocket) {
|
if (!this.websocket) {
|
||||||
// set a timer and retry?
|
// set a timer and retry?
|
||||||
@@ -223,25 +272,25 @@ class App {
|
|||||||
};
|
};
|
||||||
marked.setOptions({ renderer: renderer });
|
marked.setOptions({ renderer: renderer });
|
||||||
}
|
}
|
||||||
arrayBufferToBase64(buffer) {
|
// arrayBufferToBase64(buffer: ArrayBuffer) {
|
||||||
return new Promise((resolve, reject) => {
|
// return new Promise((resolve, reject) => {
|
||||||
const blob = new Blob([buffer], { type: 'application/octet-stream' });
|
// const blob = new Blob([buffer], { type: 'application/octet-stream' });
|
||||||
const reader = new FileReader();
|
// const reader = new FileReader();
|
||||||
reader.onloadend = () => {
|
// reader.onloadend = () => {
|
||||||
const dataUrl = reader.result;
|
// const dataUrl = reader.result as string;
|
||||||
if (!dataUrl) {
|
// if (!dataUrl) {
|
||||||
resolve(null);
|
// resolve(null);
|
||||||
return;
|
// return;
|
||||||
}
|
// }
|
||||||
const base64 = dataUrl.split(',')[1];
|
// const base64 = dataUrl.split(',')[1];
|
||||||
resolve(base64);
|
// resolve(base64);
|
||||||
};
|
// };
|
||||||
reader.onerror = (error) => {
|
// reader.onerror = (error) => {
|
||||||
reject(error);
|
// reject(error);
|
||||||
};
|
// };
|
||||||
reader.readAsDataURL(blob);
|
// reader.readAsDataURL(blob);
|
||||||
});
|
// });
|
||||||
}
|
// }
|
||||||
async createTestData() {
|
async createTestData() {
|
||||||
let postsTestData = await (await fetch("./postsTestData.json")).json();
|
let postsTestData = await (await fetch("./postsTestData.json")).json();
|
||||||
return postsTestData;
|
return postsTestData;
|
||||||
@@ -330,12 +379,12 @@ class App {
|
|||||||
console.error("Service Worker registration failed:", error);
|
console.error("Service Worker registration failed:", error);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
addPost(userID, postText) {
|
addPost(userID, postText, imageData) {
|
||||||
if ((typeof postText !== "string") || postText.length === 0) {
|
if ((typeof postText !== "string") || postText.length === 0) {
|
||||||
log("Not posting an empty string...");
|
log("Not posting an empty string...");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
let post = new Post(this.username, userID, postText, new Date());
|
let post = new Post(this.username, userID, postText, new Date(), imageData);
|
||||||
this.posts.push(post);
|
this.posts.push(post);
|
||||||
// localStorage.setItem(key, JSON.stringify(posts));
|
// localStorage.setItem(key, JSON.stringify(posts));
|
||||||
addData(userID, post);
|
addData(userID, post);
|
||||||
@@ -436,7 +485,8 @@ class App {
|
|||||||
let importTweetsButton = document.getElementById("import_tweets");
|
let importTweetsButton = document.getElementById("import_tweets");
|
||||||
let clearPostsButton = document.getElementById("clear_posts");
|
let clearPostsButton = document.getElementById("clear_posts");
|
||||||
let updateApp = document.getElementById("update_app");
|
let updateApp = document.getElementById("update_app");
|
||||||
let ddlnLogoButton = document.getElementById('ddln-logo-button');
|
let ddlnLogoButton = document.getElementById('ddln_logo_button');
|
||||||
|
// let addP = document.getElementById('button_add_pic') as HTMLDivElement;
|
||||||
let usernameField = document.getElementById('username');
|
let usernameField = document.getElementById('username');
|
||||||
usernameField?.addEventListener('input', (event) => {
|
usernameField?.addEventListener('input', (event) => {
|
||||||
this.username = event.target.innerText;
|
this.username = event.target.innerText;
|
||||||
@@ -464,6 +514,12 @@ class App {
|
|||||||
if (!(postButton && postText)) {
|
if (!(postButton && postText)) {
|
||||||
throw new Error();
|
throw new Error();
|
||||||
}
|
}
|
||||||
|
postText.addEventListener('paste', async (e) => {
|
||||||
|
const dataTransfer = e.clipboardData;
|
||||||
|
const file = dataTransfer.files[0];
|
||||||
|
let buffer = await file.arrayBuffer();
|
||||||
|
let type = this.addPost(this.userID, 'image...', buffer);
|
||||||
|
});
|
||||||
postButton.addEventListener("click", () => {
|
postButton.addEventListener("click", () => {
|
||||||
this.addPost(userID, postText.value);
|
this.addPost(userID, postText.value);
|
||||||
postText.value = "";
|
postText.value = "";
|
||||||
@@ -506,22 +562,6 @@ class App {
|
|||||||
let peerID = this.getPeerID();
|
let peerID = this.getPeerID();
|
||||||
this.userID = userID;
|
this.userID = userID;
|
||||||
this.peerID = peerID;
|
this.peerID = peerID;
|
||||||
let connectURL = `https://${document.location.hostname}?connect=${this.userID}`;
|
|
||||||
document.getElementById('connectURL').innerHTML = `<a href="${connectURL}">connect</a>`;
|
|
||||||
let qrcode = await new QRCode(document.getElementById('qrcode'), {
|
|
||||||
text: connectURL,
|
|
||||||
width: 256,
|
|
||||||
height: 256,
|
|
||||||
colorDark: "#000000",
|
|
||||||
colorLight: "#ffffff",
|
|
||||||
correctLevel: QRCode.CorrectLevel.H
|
|
||||||
});
|
|
||||||
let qrcodeImage = document.querySelector('#qrcode > img');
|
|
||||||
qrcodeImage.classList.add('qrcode_image');
|
|
||||||
log(`user:${userID} peer:${peerID}`);
|
|
||||||
let websocket = new wsConnection(userID, peerID);
|
|
||||||
window.addEventListener('beforeunload', () => { websocket.disconnect(); });
|
|
||||||
this.initOffline(websocket);
|
|
||||||
this.initButtons(userID, this.posts, registration);
|
this.initButtons(userID, this.posts, registration);
|
||||||
let time = 0;
|
let time = 0;
|
||||||
let delta = 0;
|
let delta = 0;
|
||||||
@@ -546,6 +586,22 @@ class App {
|
|||||||
if (performance?.memory) {
|
if (performance?.memory) {
|
||||||
log(`memory used: ${(performance.memory.usedJSHeapSize / 1024 / 1024).toFixed(2)}Mb`);
|
log(`memory used: ${(performance.memory.usedJSHeapSize / 1024 / 1024).toFixed(2)}Mb`);
|
||||||
}
|
}
|
||||||
|
let connectURL = `https://${document.location.hostname}?connect=${this.userID}`;
|
||||||
|
document.getElementById('connectURL').innerHTML = `<a href="${connectURL}">connect</a>`;
|
||||||
|
let qrcode = await new QRCode(document.getElementById('qrcode'), {
|
||||||
|
text: connectURL,
|
||||||
|
width: 256,
|
||||||
|
height: 256,
|
||||||
|
colorDark: "#000000",
|
||||||
|
colorLight: "#ffffff",
|
||||||
|
correctLevel: QRCode.CorrectLevel.H
|
||||||
|
});
|
||||||
|
let qrcodeImage = document.querySelector('#qrcode > img');
|
||||||
|
qrcodeImage.classList.add('qrcode_image');
|
||||||
|
log(`user:${userID} peer:${peerID}`);
|
||||||
|
let websocket = new wsConnection(userID, peerID);
|
||||||
|
window.addEventListener('beforeunload', () => { websocket.disconnect(); });
|
||||||
|
this.initOffline(websocket);
|
||||||
// const client = new WebTorrent()
|
// const client = new WebTorrent()
|
||||||
// // Sintel, a free, Creative Commons movie
|
// // Sintel, a free, Creative Commons movie
|
||||||
// const torrentId = 'magnet:?xt=urn:btih:6091e199a8d9272a40dd9a25a621a5c355d6b0be&dn=WING+IT!+-+Blender+Open+Movie+1080p.mp4&tr=wss%3A%2F%2Ftracker.btorrent.xyz&tr=wss%3A%2F%2Ftracker.openwebtorrent.com&tr=udp%3A%2F%2Ftracker.leechers-paradise.org%3A6969&tr=udp%3A%2F%2Ftracker.coppersurfer.tk%3A6969&tr=udp%3A%2F%2Ftracker.opentrackr.org%3A1337&tr=udp%3A%2F%2Fexplodie.org%3A6969&tr=udp%3A%2F%2Ftracker.empire-js.us%3A1337';
|
// const torrentId = 'magnet:?xt=urn:btih:6091e199a8d9272a40dd9a25a621a5c355d6b0be&dn=WING+IT!+-+Blender+Open+Movie+1080p.mp4&tr=wss%3A%2F%2Ftracker.btorrent.xyz&tr=wss%3A%2F%2Ftracker.openwebtorrent.com&tr=udp%3A%2F%2Ftracker.leechers-paradise.org%3A6969&tr=udp%3A%2F%2Ftracker.coppersurfer.tk%3A6969&tr=udp%3A%2F%2Ftracker.opentrackr.org%3A1337&tr=udp%3A%2F%2Fexplodie.org%3A6969&tr=udp%3A%2F%2Ftracker.empire-js.us%3A1337';
|
||||||
@@ -566,17 +622,17 @@ class App {
|
|||||||
throw new Error();
|
throw new Error();
|
||||||
}
|
}
|
||||||
contentDiv.innerHTML = "";
|
contentDiv.innerHTML = "";
|
||||||
let count = 0;
|
// let count = 0;
|
||||||
for (let i = posts.length - 1; i >= 0; i--) {
|
for (let i = posts.length - 1; i >= 0; i--) {
|
||||||
let postData = posts[i];
|
let postData = posts[i];
|
||||||
let post = this.renderPost(postData, posts);
|
let post = this.renderPost(postData, posts);
|
||||||
if (post) {
|
if (post) {
|
||||||
fragment.appendChild(post);
|
fragment.appendChild(post);
|
||||||
count++;
|
// count++;
|
||||||
}
|
|
||||||
if (count > 100) {
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
|
// if (count > 100) {
|
||||||
|
// break;
|
||||||
|
// }
|
||||||
}
|
}
|
||||||
if (!contentDiv) {
|
if (!contentDiv) {
|
||||||
throw new Error("Couldn't get content div!");
|
throw new Error("Couldn't get content div!");
|
||||||
@@ -610,21 +666,22 @@ class App {
|
|||||||
containerDiv.innerHTML = postTemplate;
|
containerDiv.innerHTML = postTemplate;
|
||||||
containerDiv.querySelector('#deleteButton')?.appendChild(deleteButton);
|
containerDiv.querySelector('#deleteButton')?.appendChild(deleteButton);
|
||||||
containerDiv.querySelector('#editButton')?.appendChild(editButton);
|
containerDiv.querySelector('#editButton')?.appendChild(editButton);
|
||||||
// if (!("image_data" in post && post.image_data)) {
|
if (!("image_data" in post && post.image_data)) {
|
||||||
// containerDiv.appendChild(timestampDiv);
|
// containerDiv.appendChild(timestampDiv);
|
||||||
// return containerDiv;
|
return containerDiv;
|
||||||
// // return null;
|
// return null;
|
||||||
// }
|
}
|
||||||
// let image = document.createElement("img");
|
let image = document.createElement("img");
|
||||||
// const blob = new Blob([post.image_data as ArrayBuffer], { type: 'image/jpg' });
|
// const blob = new Blob([post.image_data as ArrayBuffer], { type: 'image/png' });
|
||||||
// const url = URL.createObjectURL(blob);
|
const blob = new Blob([post.image_data]);
|
||||||
// image.onload = () => {
|
const url = URL.createObjectURL(blob);
|
||||||
// URL.revokeObjectURL(url);
|
image.onload = () => {
|
||||||
// };
|
URL.revokeObjectURL(url);
|
||||||
// image.src = url;
|
};
|
||||||
// // image.src = image.src = "data:image/png;base64," + post.image;
|
image.src = url;
|
||||||
// image.className = "postImage";
|
// image.src = image.src = "data:image/png;base64," + post.image;
|
||||||
// containerDiv.appendChild(image);
|
image.className = "postImage";
|
||||||
|
containerDiv.appendChild(image);
|
||||||
// containerDiv.appendChild(timestampDiv);
|
// containerDiv.appendChild(timestampDiv);
|
||||||
return containerDiv;
|
return containerDiv;
|
||||||
}
|
}
|
||||||
|
|||||||
File diff suppressed because one or more lines are too long
113
src/db.ts
113
src/db.ts
@@ -159,7 +159,7 @@ export async function addDataArray(userID: string, array: any[]): Promise<void>
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function mergeDataArray(userID: string, array:any[]): Promise<void> {
|
export async function checkPostIds(userID: string, post_ids: string[]) {
|
||||||
try {
|
try {
|
||||||
const db = await openDatabase(userID);
|
const db = await openDatabase(userID);
|
||||||
const transaction = db.transaction(postStoreName, "readwrite");
|
const transaction = db.transaction(postStoreName, "readwrite");
|
||||||
@@ -176,7 +176,60 @@ export async function mergeDataArray(userID: string, array:any[]): Promise<void>
|
|||||||
db.close();
|
db.close();
|
||||||
};
|
};
|
||||||
|
|
||||||
let postsToWrite:any = [];
|
let postIdsNeeded: any = [];
|
||||||
|
|
||||||
|
for (let id of post_ids) {
|
||||||
|
try {
|
||||||
|
let havePost = await new Promise<boolean>((resolve, reject) => {
|
||||||
|
const getRequest = index.getKey(id);
|
||||||
|
|
||||||
|
getRequest.onerror = (e) => {
|
||||||
|
console.log((e.target as IDBRequest).error);
|
||||||
|
reject(e);
|
||||||
|
};
|
||||||
|
|
||||||
|
getRequest.onsuccess = async (e) => {
|
||||||
|
const key = (e.target as IDBRequest).result;
|
||||||
|
resolve(key !== undefined)
|
||||||
|
};
|
||||||
|
|
||||||
|
});
|
||||||
|
|
||||||
|
// console.log(post.post_id, havePost);
|
||||||
|
if (!havePost) {
|
||||||
|
postIdsNeeded.push(id);
|
||||||
|
}
|
||||||
|
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Error processing post:", error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
console.log(`checkPostIds need ${postIdsNeeded.length} posts`);
|
||||||
|
return postIdsNeeded;
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Error in opening database:", error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function mergeDataArray(userID: string, array: any[]): Promise<void> {
|
||||||
|
try {
|
||||||
|
const db = await openDatabase(userID);
|
||||||
|
const transaction = db.transaction(postStoreName, "readwrite");
|
||||||
|
const store = transaction.objectStore(postStoreName);
|
||||||
|
const index = store.index("postIDIndex");
|
||||||
|
|
||||||
|
transaction.oncomplete = () => {
|
||||||
|
console.log("Transaction completed successfully");
|
||||||
|
db.close();
|
||||||
|
};
|
||||||
|
|
||||||
|
transaction.onerror = (event) => {
|
||||||
|
console.error("Transaction error:", (event.target as any).error);
|
||||||
|
db.close();
|
||||||
|
};
|
||||||
|
|
||||||
|
let postsToWrite: any = [];
|
||||||
|
|
||||||
for (let post of array) {
|
for (let post of array) {
|
||||||
try {
|
try {
|
||||||
@@ -196,7 +249,7 @@ export async function mergeDataArray(userID: string, array:any[]): Promise<void>
|
|||||||
});
|
});
|
||||||
|
|
||||||
// console.log(post.post_id, havePost);
|
// console.log(post.post_id, havePost);
|
||||||
if (!havePost ) {
|
if (!havePost) {
|
||||||
postsToWrite.push(post);
|
postsToWrite.push(post);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -276,3 +329,57 @@ export async function getAllData(userID: string): Promise<any | undefined> {
|
|||||||
};
|
};
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export async function getAllIds(userID: string): Promise<any | undefined> {
|
||||||
|
const db = await openDatabase(userID);
|
||||||
|
const transaction = db.transaction(postStoreName, "readonly");
|
||||||
|
const store = transaction.objectStore(postStoreName);
|
||||||
|
const index = store.index("postIDIndex");
|
||||||
|
|
||||||
|
let keys: string[] = [];
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
let request = index.openKeyCursor();
|
||||||
|
|
||||||
|
request.onsuccess = (event:any) => {
|
||||||
|
let cursor = event.target.result;
|
||||||
|
if (cursor) {
|
||||||
|
keys.push(cursor.key);
|
||||||
|
cursor.continue();
|
||||||
|
} else {
|
||||||
|
resolve(keys);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
request.onerror = (event: any) => {
|
||||||
|
reject(event);
|
||||||
|
};
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function getPostsByIds(userID:string, postIDs:string[]) {
|
||||||
|
const db = await openDatabase(userID);
|
||||||
|
const transaction = db.transaction(postStoreName, "readonly");
|
||||||
|
const store = transaction.objectStore(postStoreName);
|
||||||
|
const index = store.index("postIDIndex");
|
||||||
|
let posts = [];
|
||||||
|
|
||||||
|
for (const postID of postIDs) {
|
||||||
|
const post = await new Promise((resolve, reject) => {
|
||||||
|
let request = index.get(postID);
|
||||||
|
|
||||||
|
request.onsuccess = (event:any) => {
|
||||||
|
resolve(event.target.result); // Resolve with the post
|
||||||
|
};
|
||||||
|
|
||||||
|
request.onerror = (event) => {
|
||||||
|
reject(event); // Reject if any error occurs
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
if (post) {
|
||||||
|
posts.push(post); // Add the post to the result array if found
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return posts; // Return the array of posts
|
||||||
|
}
|
||||||
247
src/main.ts
247
src/main.ts
@@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
import { openDatabase, getData, addData, addDataArray, clearData, deleteData, mergeDataArray, getAllData } from "./db.js"
|
import { getData, addData, addDataArray, clearData, deleteData, mergeDataArray, getAllData, checkPostIds, getAllIds, getPostsByIds} from "./db.js"
|
||||||
|
|
||||||
declare let WebTorrent: any;
|
declare let WebTorrent: any;
|
||||||
|
|
||||||
@@ -67,7 +67,15 @@ class Post {
|
|||||||
importedFrom: "twitter" | null;
|
importedFrom: "twitter" | null;
|
||||||
importSource: any;
|
importSource: any;
|
||||||
|
|
||||||
constructor(author: string, author_id: string, text: string, post_timestamp: Date, imageData: ArrayBuffer | null = null, importedFrom: "twitter" | null = null, importSource: any = null) {
|
constructor(
|
||||||
|
author: string,
|
||||||
|
author_id: string,
|
||||||
|
text: string,
|
||||||
|
post_timestamp: Date,
|
||||||
|
imageData: ArrayBuffer | null = null,
|
||||||
|
importedFrom: "twitter" | null = null,
|
||||||
|
importSource: any = null) {
|
||||||
|
|
||||||
this.post_timestamp = post_timestamp;
|
this.post_timestamp = post_timestamp;
|
||||||
this.post_id = generateID();
|
this.post_id = generateID();
|
||||||
|
|
||||||
@@ -92,8 +100,8 @@ window.addEventListener('scroll', () => {
|
|||||||
if (scrollPoint >= totalPageHeight) {
|
if (scrollPoint >= totalPageHeight) {
|
||||||
console.log('Scrolled to the bottom!');
|
console.log('Scrolled to the bottom!');
|
||||||
console.log(scrollPoint, totalPageHeight);
|
console.log(scrollPoint, totalPageHeight);
|
||||||
// You can perform your action here
|
|
||||||
}
|
}
|
||||||
|
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
||||||
@@ -115,6 +123,25 @@ window.addEventListener('scroll', () => {
|
|||||||
// }
|
// }
|
||||||
// }
|
// }
|
||||||
|
|
||||||
|
function arrayBufferToBase64( buffer:ArrayBuffer ) {
|
||||||
|
var binary = '';
|
||||||
|
var bytes = new Uint8Array( buffer );
|
||||||
|
var len = bytes.byteLength;
|
||||||
|
for (var i = 0; i < len; i++) {
|
||||||
|
binary += String.fromCharCode( bytes[ i ] );
|
||||||
|
}
|
||||||
|
return window.btoa( binary );
|
||||||
|
}
|
||||||
|
|
||||||
|
function base64ToArrayBuffer(base64:string) {
|
||||||
|
var binaryString = atob(base64);
|
||||||
|
var bytes = new Uint8Array(binaryString.length);
|
||||||
|
for (var i = 0; i < binaryString.length; i++) {
|
||||||
|
bytes[i] = binaryString.charCodeAt(i);
|
||||||
|
}
|
||||||
|
return bytes.buffer;
|
||||||
|
}
|
||||||
|
|
||||||
class wsConnection {
|
class wsConnection {
|
||||||
websocket: WebSocket | null = null;
|
websocket: WebSocket | null = null;
|
||||||
userID = "";
|
userID = "";
|
||||||
@@ -135,7 +162,7 @@ class wsConnection {
|
|||||||
} catch (e) {
|
} catch (e) {
|
||||||
console.log(e, "wsConnection send: Couldn't serialize message", message);
|
console.log(e, "wsConnection send: Couldn't serialize message", message);
|
||||||
}
|
}
|
||||||
log(`ws->${json.slice(0,80)}`)
|
log(`ws->${json.slice(0,240)}`)
|
||||||
this.websocket!.send(json);
|
this.websocket!.send(json);
|
||||||
|
|
||||||
}
|
}
|
||||||
@@ -153,7 +180,7 @@ class wsConnection {
|
|||||||
type:"peer_message",
|
type:"peer_message",
|
||||||
from:this.peerID,
|
from:this.peerID,
|
||||||
to:peerID,
|
to:peerID,
|
||||||
message:{type:"get_posts_for_user", user_id:userID}
|
message:{type:"get_post_ids_for_user", user_id:userID}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -162,13 +189,63 @@ class wsConnection {
|
|||||||
pongHandler(data: any) {
|
pongHandler(data: any) {
|
||||||
}
|
}
|
||||||
|
|
||||||
async getPostsForUserResponseHandler(data: any) {
|
async getPostIdsForUserResponseHandler(data: any) {
|
||||||
// log(`getPostsForUserResponse: ${data}`)
|
// log(`getPostsForUserResponse: ${data}`)
|
||||||
|
|
||||||
|
let message = data.message;
|
||||||
|
console.log(`getPostIdsForUserResponseHandler Got ${message.post_ids.length} from peer ${data.from}`);
|
||||||
|
|
||||||
|
console.log(`Checking post IDs...`);
|
||||||
|
|
||||||
|
let postIds = await checkPostIds(message.user_id, data.message.post_ids);
|
||||||
|
if (postIds.length === 0) {
|
||||||
|
log(`Don't need any posts from peer ${data.from}`);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
let responseMessage = { type: "peer_message", from: app.peerID, to: data.from, message: { type: "get_posts_for_user", post_ids: postIds, user_id: message.user_id } }
|
||||||
|
|
||||||
|
this.send(responseMessage);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
async getPostIdsForUserHandler(data: any) {
|
||||||
|
let message = data.message;
|
||||||
|
let postIds = await getAllIds(message.user_id) ?? [];
|
||||||
|
let responseMessage = { type: "peer_message", from: app.peerID, to: data.from, message: { type: "get_post_ids_for_user_response", post_ids: postIds, user_id: message.user_id } }
|
||||||
|
this.send(responseMessage)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Send posts to peer
|
||||||
|
async getPostsForUserHandler(data: any) {
|
||||||
|
let message = data.message;
|
||||||
|
let posts = await getPostsByIds(message.user_id, message.post_ids) ?? [];
|
||||||
|
|
||||||
|
let output = [];
|
||||||
|
for (let post of posts) {
|
||||||
|
let newPost = (post as any).data; if (newPost.image_data) {
|
||||||
|
newPost.image_data = arrayBufferToBase64(newPost.image_data)
|
||||||
|
}
|
||||||
|
|
||||||
|
output.push(newPost);
|
||||||
|
}
|
||||||
|
|
||||||
|
// posts = posts.map((post:any)=>{let newPost = post.data; if (newPost.image_data){newPost.image_data = arraybufferto};return newPost});
|
||||||
|
// posts = posts.map((post:any)=>{})
|
||||||
|
let responseMessage = { type: "peer_message", from: app.peerID, to: data.from, message: { type: "get_posts_for_user_response", posts: output, user_id: message.user_id } }
|
||||||
|
this.send(responseMessage)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Got posts from peer
|
||||||
|
async getPostsForUserReponseHandler(data: any) {
|
||||||
let message = data.message;
|
let message = data.message;
|
||||||
console.log(`getPostsForUserResponseHandler Got ${message.posts.length} from peer ${data.from}`);
|
console.log(`getPostsForUserResponseHandler Got ${message.posts.length} from peer ${data.from}`);
|
||||||
for (let post of message.posts) {
|
for (let post of message.posts) {
|
||||||
post.post_timestamp = new Date(post.post_timestamp);
|
post.post_timestamp = new Date(post.post_timestamp);
|
||||||
|
if (post.image_data) {
|
||||||
|
post.image_data = base64ToArrayBuffer(post.image_data);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
console.log(`Merging same user peer posts...`)
|
console.log(`Merging same user peer posts...`)
|
||||||
await mergeDataArray(message.user_id, data.message.posts)
|
await mergeDataArray(message.user_id, data.message.posts)
|
||||||
@@ -180,16 +257,6 @@ class wsConnection {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async getPostsForUserHandler(data: any) {
|
|
||||||
let message = data.message;
|
|
||||||
let posts = await getAllData(message.user_id) ?? []; // this doesn't get all posts!
|
|
||||||
posts = posts.map((post:any)=>post.data)
|
|
||||||
// let posts = await getAllData(message.user_id) ?? [];
|
|
||||||
|
|
||||||
let responseMessage = { type: "peer_message", from: app.peerID, to: data.from, message: { type: "get_posts_for_user_response", posts: posts, user_id: message.user_id } }
|
|
||||||
this.send(responseMessage)
|
|
||||||
}
|
|
||||||
|
|
||||||
async peerMessageHandler(data: any) {
|
async peerMessageHandler(data: any) {
|
||||||
log(`peerMessageHandler ${data}`)
|
log(`peerMessageHandler ${data}`)
|
||||||
|
|
||||||
@@ -239,7 +306,7 @@ class wsConnection {
|
|||||||
};
|
};
|
||||||
|
|
||||||
this.websocket.onmessage = (event) => {
|
this.websocket.onmessage = (event) => {
|
||||||
log('ws:<-' + event.data.slice(0,80));
|
log('ws:<-' + event.data.slice(0,240));
|
||||||
|
|
||||||
let data = JSON.parse(event.data);
|
let data = JSON.parse(event.data);
|
||||||
|
|
||||||
@@ -273,8 +340,13 @@ class wsConnection {
|
|||||||
this.messageHandlers.set('pong', this.pongHandler);
|
this.messageHandlers.set('pong', this.pongHandler);
|
||||||
this.messageHandlers.set('peer_message', this.peerMessageHandler.bind(this));
|
this.messageHandlers.set('peer_message', this.peerMessageHandler.bind(this));
|
||||||
|
|
||||||
|
this.peerMessageHandlers.set('get_post_ids_for_user', this.getPostIdsForUserHandler.bind(this));
|
||||||
|
this.peerMessageHandlers.set('get_post_ids_for_user_response', this.getPostIdsForUserResponseHandler.bind(this));
|
||||||
|
|
||||||
this.peerMessageHandlers.set('get_posts_for_user', this.getPostsForUserHandler.bind(this));
|
this.peerMessageHandlers.set('get_posts_for_user', this.getPostsForUserHandler.bind(this));
|
||||||
this.peerMessageHandlers.set('get_posts_for_user_response', this.getPostsForUserResponseHandler.bind(this));
|
this.peerMessageHandlers.set('get_posts_for_user_response', this.getPostsForUserReponseHandler.bind(this));
|
||||||
|
|
||||||
|
|
||||||
this.connect();
|
this.connect();
|
||||||
|
|
||||||
if (!this.websocket) {
|
if (!this.websocket) {
|
||||||
@@ -298,28 +370,28 @@ class App {
|
|||||||
marked.setOptions({ renderer: renderer });
|
marked.setOptions({ renderer: renderer });
|
||||||
}
|
}
|
||||||
|
|
||||||
arrayBufferToBase64(buffer: ArrayBuffer) {
|
// arrayBufferToBase64(buffer: ArrayBuffer) {
|
||||||
return new Promise((resolve, reject) => {
|
// return new Promise((resolve, reject) => {
|
||||||
const blob = new Blob([buffer], { type: 'application/octet-stream' });
|
// const blob = new Blob([buffer], { type: 'application/octet-stream' });
|
||||||
const reader = new FileReader();
|
// const reader = new FileReader();
|
||||||
|
|
||||||
reader.onloadend = () => {
|
// reader.onloadend = () => {
|
||||||
const dataUrl = reader.result as string;
|
// const dataUrl = reader.result as string;
|
||||||
if (!dataUrl) {
|
// if (!dataUrl) {
|
||||||
resolve(null);
|
// resolve(null);
|
||||||
return;
|
// return;
|
||||||
}
|
// }
|
||||||
const base64 = dataUrl.split(',')[1];
|
// const base64 = dataUrl.split(',')[1];
|
||||||
resolve(base64);
|
// resolve(base64);
|
||||||
};
|
// };
|
||||||
|
|
||||||
reader.onerror = (error) => {
|
// reader.onerror = (error) => {
|
||||||
reject(error);
|
// reject(error);
|
||||||
};
|
// };
|
||||||
|
|
||||||
reader.readAsDataURL(blob);
|
// reader.readAsDataURL(blob);
|
||||||
});
|
// });
|
||||||
}
|
// }
|
||||||
|
|
||||||
async createTestData() {
|
async createTestData() {
|
||||||
let postsTestData = await (await fetch("./postsTestData.json")).json();
|
let postsTestData = await (await fetch("./postsTestData.json")).json();
|
||||||
@@ -439,13 +511,13 @@ class App {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
addPost(userID: string, postText: string) {
|
addPost(userID: string, postText: string, imageData?: ArrayBuffer) {
|
||||||
if ((typeof postText !== "string") || postText.length === 0) {
|
if ((typeof postText !== "string") || postText.length === 0) {
|
||||||
log("Not posting an empty string...")
|
log("Not posting an empty string...")
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
let post = new Post(this.username, userID, postText, new Date());
|
let post = new Post(this.username, userID, postText, new Date(), imageData);
|
||||||
|
|
||||||
this.posts.push(post);
|
this.posts.push(post);
|
||||||
// localStorage.setItem(key, JSON.stringify(posts));
|
// localStorage.setItem(key, JSON.stringify(posts));
|
||||||
@@ -455,6 +527,8 @@ class App {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
getPeerID() {
|
getPeerID() {
|
||||||
let id = localStorage.getItem("peer_id");
|
let id = localStorage.getItem("peer_id");
|
||||||
|
|
||||||
@@ -575,7 +649,12 @@ class App {
|
|||||||
let importTweetsButton = document.getElementById("import_tweets") as HTMLButtonElement;
|
let importTweetsButton = document.getElementById("import_tweets") as HTMLButtonElement;
|
||||||
let clearPostsButton = document.getElementById("clear_posts") as HTMLButtonElement;
|
let clearPostsButton = document.getElementById("clear_posts") as HTMLButtonElement;
|
||||||
let updateApp = document.getElementById("update_app") as HTMLButtonElement;
|
let updateApp = document.getElementById("update_app") as HTMLButtonElement;
|
||||||
let ddlnLogoButton = document.getElementById('ddln-logo-button') as HTMLDivElement;
|
let ddlnLogoButton = document.getElementById('ddln_logo_button') as HTMLDivElement;
|
||||||
|
// let addP = document.getElementById('button_add_pic') as HTMLDivElement;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
let usernameField = document.getElementById('username');
|
let usernameField = document.getElementById('username');
|
||||||
usernameField?.addEventListener('input', (event:any)=>{
|
usernameField?.addEventListener('input', (event:any)=>{
|
||||||
@@ -614,6 +693,14 @@ class App {
|
|||||||
throw new Error();
|
throw new Error();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
postText.addEventListener('paste', async (e)=>{
|
||||||
|
const dataTransfer = e.clipboardData
|
||||||
|
const file = dataTransfer!.files[ 0 ];
|
||||||
|
let buffer = await file.arrayBuffer();
|
||||||
|
let type =
|
||||||
|
this.addPost(this.userID, 'image...', buffer);
|
||||||
|
});
|
||||||
|
|
||||||
postButton.addEventListener("click", () => {
|
postButton.addEventListener("click", () => {
|
||||||
this.addPost(userID, postText.value);
|
this.addPost(userID, postText.value);
|
||||||
postText.value = "";
|
postText.value = "";
|
||||||
@@ -669,27 +756,6 @@ class App {
|
|||||||
let peerID = this.getPeerID();
|
let peerID = this.getPeerID();
|
||||||
this.userID = userID;
|
this.userID = userID;
|
||||||
this.peerID = peerID;
|
this.peerID = peerID;
|
||||||
|
|
||||||
let connectURL = `https://${document.location.hostname}?connect=${this.userID}`;
|
|
||||||
document.getElementById('connectURL')!.innerHTML = `<a href="${connectURL}">connect</a>`;
|
|
||||||
|
|
||||||
let qrcode = await new QRCode(document.getElementById('qrcode'), {
|
|
||||||
text: connectURL,
|
|
||||||
width: 256,
|
|
||||||
height: 256,
|
|
||||||
colorDark: "#000000",
|
|
||||||
colorLight: "#ffffff",
|
|
||||||
correctLevel: QRCode.CorrectLevel.H
|
|
||||||
});
|
|
||||||
|
|
||||||
|
|
||||||
let qrcodeImage:HTMLImageElement = document.querySelector('#qrcode > img') as HTMLImageElement;
|
|
||||||
qrcodeImage.classList.add('qrcode_image');
|
|
||||||
|
|
||||||
log(`user:${userID} peer:${peerID}`);
|
|
||||||
let websocket = new wsConnection(userID, peerID);
|
|
||||||
window.addEventListener('beforeunload', () => { websocket.disconnect() })
|
|
||||||
this.initOffline(websocket);
|
|
||||||
this.initButtons(userID, this.posts, registration);
|
this.initButtons(userID, this.posts, registration);
|
||||||
|
|
||||||
let time = 0;
|
let time = 0;
|
||||||
@@ -723,6 +789,26 @@ class App {
|
|||||||
if ((performance as any)?.memory) {
|
if ((performance as any)?.memory) {
|
||||||
log(`memory used: ${((performance as any).memory.usedJSHeapSize / 1024 / 1024).toFixed(2)}Mb`)
|
log(`memory used: ${((performance as any).memory.usedJSHeapSize / 1024 / 1024).toFixed(2)}Mb`)
|
||||||
}
|
}
|
||||||
|
let connectURL = `https://${document.location.hostname}?connect=${this.userID}`;
|
||||||
|
document.getElementById('connectURL')!.innerHTML = `<a href="${connectURL}">connect</a>`;
|
||||||
|
|
||||||
|
let qrcode = await new QRCode(document.getElementById('qrcode'), {
|
||||||
|
text: connectURL,
|
||||||
|
width: 256,
|
||||||
|
height: 256,
|
||||||
|
colorDark: "#000000",
|
||||||
|
colorLight: "#ffffff",
|
||||||
|
correctLevel: QRCode.CorrectLevel.H
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
|
let qrcodeImage:HTMLImageElement = document.querySelector('#qrcode > img') as HTMLImageElement;
|
||||||
|
qrcodeImage.classList.add('qrcode_image');
|
||||||
|
|
||||||
|
log(`user:${userID} peer:${peerID}`);
|
||||||
|
let websocket = new wsConnection(userID, peerID);
|
||||||
|
window.addEventListener('beforeunload', () => { websocket.disconnect() })
|
||||||
|
this.initOffline(websocket);
|
||||||
|
|
||||||
|
|
||||||
// const client = new WebTorrent()
|
// const client = new WebTorrent()
|
||||||
@@ -749,7 +835,7 @@ class App {
|
|||||||
throw new Error();
|
throw new Error();
|
||||||
}
|
}
|
||||||
contentDiv.innerHTML = "";
|
contentDiv.innerHTML = "";
|
||||||
let count = 0;
|
// let count = 0;
|
||||||
|
|
||||||
for (let i = posts.length - 1; i >= 0; i--) {
|
for (let i = posts.length - 1; i >= 0; i--) {
|
||||||
let postData = posts[i];
|
let postData = posts[i];
|
||||||
@@ -758,11 +844,11 @@ class App {
|
|||||||
|
|
||||||
if (post) {
|
if (post) {
|
||||||
fragment.appendChild(post);
|
fragment.appendChild(post);
|
||||||
count++;
|
// count++;
|
||||||
}
|
|
||||||
if (count > 100) {
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
|
// if (count > 100) {
|
||||||
|
// break;
|
||||||
|
// }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -807,27 +893,28 @@ class App {
|
|||||||
containerDiv.querySelector('#deleteButton')?.appendChild(deleteButton);
|
containerDiv.querySelector('#deleteButton')?.appendChild(deleteButton);
|
||||||
containerDiv.querySelector('#editButton')?.appendChild(editButton);
|
containerDiv.querySelector('#editButton')?.appendChild(editButton);
|
||||||
|
|
||||||
// if (!("image_data" in post && post.image_data)) {
|
if (!("image_data" in post && post.image_data)) {
|
||||||
// containerDiv.appendChild(timestampDiv);
|
// containerDiv.appendChild(timestampDiv);
|
||||||
// return containerDiv;
|
return containerDiv;
|
||||||
// // return null;
|
// return null;
|
||||||
// }
|
}
|
||||||
|
|
||||||
// let image = document.createElement("img");
|
let image = document.createElement("img");
|
||||||
// const blob = new Blob([post.image_data as ArrayBuffer], { type: 'image/jpg' });
|
// const blob = new Blob([post.image_data as ArrayBuffer], { type: 'image/png' });
|
||||||
// const url = URL.createObjectURL(blob);
|
const blob = new Blob([post.image_data as ArrayBuffer]);
|
||||||
// image.onload = () => {
|
const url = URL.createObjectURL(blob);
|
||||||
// URL.revokeObjectURL(url);
|
image.onload = () => {
|
||||||
// };
|
URL.revokeObjectURL(url);
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
// image.src = url;
|
image.src = url;
|
||||||
// // image.src = image.src = "data:image/png;base64," + post.image;
|
// image.src = image.src = "data:image/png;base64," + post.image;
|
||||||
// image.className = "postImage";
|
image.className = "postImage";
|
||||||
|
|
||||||
// containerDiv.appendChild(image);
|
containerDiv.appendChild(image);
|
||||||
// containerDiv.appendChild(timestampDiv);
|
// containerDiv.appendChild(timestampDiv);
|
||||||
|
|
||||||
return containerDiv;
|
return containerDiv;
|
||||||
|
|||||||
24
watch.sh
Executable file
24
watch.sh
Executable file
@@ -0,0 +1,24 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
# Check if the correct number of arguments is passed
|
||||||
|
if [ "$#" -ne 3 ]; then
|
||||||
|
echo "Usage: $0 <file_to_watch> <process_name> <command_to_run>"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Assign arguments to variables
|
||||||
|
FILE_TO_WATCH=$1
|
||||||
|
PROCESS_NAME=$2
|
||||||
|
COMMAND_TO_RUN=$3
|
||||||
|
|
||||||
|
eval "$COMMAND_TO_RUN" &
|
||||||
|
|
||||||
|
|
||||||
|
# Watch the file for changes
|
||||||
|
while inotifywait -e modify "$FILE_TO_WATCH"; do
|
||||||
|
# Kill the process
|
||||||
|
sudo pkill "$PROCESS_NAME"
|
||||||
|
|
||||||
|
# Run the provided command (like build and run)
|
||||||
|
eval "$COMMAND_TO_RUN" &
|
||||||
|
done
|
||||||
Reference in New Issue
Block a user