200 lines
5.3 KiB
Python
200 lines
5.3 KiB
Python
from orm import User
|
|
from orm.base import local_session
|
|
|
|
from resolvers.base import mutation, query, subscription
|
|
|
|
from auth.authenticate import login_required
|
|
|
|
import asyncio, uuid, json
|
|
from datetime import datetime
|
|
|
|
from redis import redis
|
|
|
|
class MessageSubscription:
|
|
queue = asyncio.Queue()
|
|
|
|
def __init__(self, chat_id):
|
|
self.chat_id = chat_id
|
|
|
|
class MessageSubscriptions:
|
|
lock = asyncio.Lock()
|
|
subscriptions = []
|
|
|
|
@staticmethod
|
|
async def register_subscription(subs):
|
|
async with MessageSubscriptions.lock:
|
|
MessageSubscriptions.subscriptions.append(subs)
|
|
|
|
@staticmethod
|
|
async def del_subscription(subs):
|
|
async with MessageSubscriptions.lock:
|
|
MessageSubscriptions.subscriptions.remove(subs)
|
|
|
|
@staticmethod
|
|
async def put(message_result):
|
|
async with MessageSubscriptions.lock:
|
|
for subs in MessageSubscriptions.subscriptions:
|
|
if message_result.message["chatId"] == subs.chat_id:
|
|
subs.queue.put_nowait(message_result)
|
|
|
|
class MessageResult:
|
|
def __init__(self, status, message):
|
|
self.status = status
|
|
self.message = message
|
|
|
|
@mutation.field("createChat")
|
|
@login_required
|
|
async def create_chat(_, info, description):
|
|
user = info.context["request"].user
|
|
|
|
chat_id = uuid.uuid4()
|
|
chat = {
|
|
"description" : description,
|
|
"createdAt" : str(datetime.now),
|
|
"createdBy" : user.slug,
|
|
"id" : str(chat_id)
|
|
}
|
|
|
|
await redis.execute("SET", f"chats/{chat_id}", json.dumps(chat))
|
|
await redis.execute("SET", f"chats/{chat_id}/next_message_id", 0)
|
|
|
|
return { "chatId" : chat_id }
|
|
|
|
async def load_messages(chatId, size, page):
|
|
message_ids = await redis.lrange(f"chats/{chatId}/message_ids",
|
|
size * (page -1), size * page - 1)
|
|
messages = []
|
|
if message_ids:
|
|
message_keys = [f"chats/{chatId}/messages/{id.decode('UTF-8')}" for id in message_ids]
|
|
messages = await redis.mget(*message_keys)
|
|
messages = [json.loads(msg) for msg in messages]
|
|
return messages
|
|
|
|
@query.field("enterChat")
|
|
@login_required
|
|
async def enter_chat(_, info, chatId, size):
|
|
chat = await redis.execute("GET", f"chats/{chatId}")
|
|
if not chat:
|
|
return { "error" : "chat not exist" }
|
|
chat = json.loads(chat)
|
|
|
|
messages = await load_messages(chatId, size, 1)
|
|
|
|
return {
|
|
"chat" : chat,
|
|
"messages" : messages
|
|
}
|
|
|
|
@mutation.field("createMessage")
|
|
@login_required
|
|
async def create_message(_, info, chatId, body, replyTo = None):
|
|
user = info.context["request"].user
|
|
|
|
chat = await redis.execute("GET", f"chats/{chatId}")
|
|
if not chat:
|
|
return { "error" : "chat not exist" }
|
|
|
|
message_id = await redis.execute("GET", f"chats/{chatId}/next_message_id")
|
|
message_id = int(message_id)
|
|
|
|
new_message = {
|
|
"chatId" : chatId,
|
|
"id" : message_id,
|
|
"author" : user.slug,
|
|
"body" : body,
|
|
"replyTo" : replyTo,
|
|
"createdAt" : datetime.now().isoformat()
|
|
}
|
|
|
|
await redis.execute("SET", f"chats/{chatId}/messages/{message_id}", json.dumps(new_message))
|
|
await redis.execute("LPUSH", f"chats/{chatId}/message_ids", str(message_id))
|
|
await redis.execute("SET", f"chats/{chatId}/next_message_id", str(message_id + 1))
|
|
|
|
result = MessageResult("NEW", new_message)
|
|
await MessageSubscriptions.put(result)
|
|
|
|
return {"message" : new_message}
|
|
|
|
@query.field("getMessages")
|
|
@login_required
|
|
async def get_messages(_, info, chatId, size, page):
|
|
chat = await redis.execute("GET", f"chats/{chatId}")
|
|
if not chat:
|
|
return { "error" : "chat not exist" }
|
|
|
|
messages = await load_messages(chatId, size, page)
|
|
|
|
return messages
|
|
|
|
@mutation.field("updateMessage")
|
|
@login_required
|
|
async def update_message(_, info, chatId, id, body):
|
|
user = info.context["request"].user
|
|
|
|
chat = await redis.execute("GET", f"chats/{chatId}")
|
|
if not chat:
|
|
return { "error" : "chat not exist" }
|
|
|
|
message = await redis.execute("GET", f"chats/{chatId}/messages/{id}")
|
|
if not message:
|
|
return { "error" : "message not exist" }
|
|
|
|
message = json.loads(message)
|
|
if message["author"] != user.slug:
|
|
return { "error" : "access denied" }
|
|
|
|
message["body"] = body
|
|
message["updatedAt"] = datetime.now().isoformat()
|
|
|
|
await redis.execute("SET", f"chats/{chatId}/messages/{id}", json.dumps(message))
|
|
|
|
result = MessageResult("UPDATED", message)
|
|
await MessageSubscriptions.put(result)
|
|
|
|
return {"message" : message}
|
|
|
|
@mutation.field("deleteMessage")
|
|
@login_required
|
|
async def delete_message(_, info, chatId, id):
|
|
user = info.context["request"].user
|
|
|
|
chat = await redis.execute("GET", f"chats/{chatId}")
|
|
if not chat:
|
|
return { "error" : "chat not exist" }
|
|
|
|
message = await redis.execute("GET", f"chats/{chatId}/messages/{id}")
|
|
if not message:
|
|
return { "error" : "message not exist" }
|
|
message = json.loads(message)
|
|
if message["author"] != user.slug:
|
|
return { "error" : "access denied" }
|
|
|
|
await redis.execute("LREM", f"chats/{chatId}/message_ids", 0, str(id))
|
|
await redis.execute("DEL", f"chats/{chatId}/messages/{id}")
|
|
|
|
result = MessageResult("DELETED", message)
|
|
await MessageSubscriptions.put(result)
|
|
|
|
return {}
|
|
|
|
@subscription.source("chatUpdated")
|
|
async def message_generator(obj, info, chatId):
|
|
|
|
#TODO: send AUTH header
|
|
#auth = info.context["request"].auth
|
|
#if not auth.logged_in:
|
|
# yield {"error" : auth.error_message or "Please login"}
|
|
|
|
try:
|
|
subs = MessageSubscription(chatId)
|
|
await MessageSubscriptions.register_subscription(subs)
|
|
while True:
|
|
msg = await subs.queue.get()
|
|
yield msg
|
|
finally:
|
|
await MessageSubscriptions.del_subscription(subs)
|
|
|
|
@subscription.field("chatUpdated")
|
|
def message_resolver(message, info, chatId):
|
|
return message
|