cache slack team users + lazy loading of channel messages

slack-api
Patrick Cleavelin 2024-04-23 22:35:25 -05:00
parent 12d2d7263e
commit ba620e66aa
6 changed files with 239 additions and 41 deletions

View File

@ -28,6 +28,7 @@ layout(std430, binding = 1) readonly buffer GlyphBlock {
layout(std430, binding = 2) readonly buffer ParamsBlock { layout(std430, binding = 2) readonly buffer ParamsBlock {
vec2 screen_size; vec2 screen_size;
vec2 font_size;
}; };
vec4 to_device_position(vec2 position, vec2 size) { vec4 to_device_position(vec2 position, vec2 size) {
@ -41,7 +42,7 @@ void main() {
vec2 scaled_size = ((vertices[gl_VertexID].position + 1.0) / 2.0) * (glyph.size/2.0); vec2 scaled_size = ((vertices[gl_VertexID].position + 1.0) / 2.0) * (glyph.size/2.0);
vec2 scaled_size_2 = ((vertices[gl_VertexID].position + 1.0) / 2.0) * (glyph.size); vec2 scaled_size_2 = ((vertices[gl_VertexID].position + 1.0) / 2.0) * (glyph.size);
vec2 glyph_pos = scaled_size + glyph.target_position + vec2(0, glyph.y_offset/2.0+24); vec2 glyph_pos = scaled_size + glyph.target_position + vec2(0, glyph.y_offset/2.0+font_size.y);
vec4 device_position = to_device_position(glyph_pos, screen_size); vec4 device_position = to_device_position(glyph_pos, screen_size);
vec2 atlas_position = (scaled_size_2 + glyph.atlas_position) / 512.0; vec2 atlas_position = (scaled_size_2 + glyph.atlas_position) / 512.0;

View File

@ -52,6 +52,7 @@ arrayTemplate(GpuGlyph);
typedef struct { typedef struct {
float screen_size[2]; float screen_size[2];
float font_size[2];
} GpuUniformParams; } GpuUniformParams;
#if defined(__APPLE__) #if defined(__APPLE__)
@ -495,6 +496,11 @@ static void _metal_gfx_present(_metal_gfx_context *cx) {
(float)_gfx_context.frame_width, (float)_gfx_context.frame_width,
(float)_gfx_context.frame_height, (float)_gfx_context.frame_height,
}, },
.font_size =
{
(float)_FONT_WIDTH,
(float)_FONT_HEIGHT,
},
}; };
gfx_update_buffer(&_gfx_context, 3, &gpu_uniform_params, gfx_update_buffer(&_gfx_context, 3, &gpu_uniform_params,
@ -641,12 +647,12 @@ static void _metal_gfx_update_buffer(_metal_gfx_context *cx,
static void _wayland_pointer_enter(void *data, struct wl_pointer *pointer, static void _wayland_pointer_enter(void *data, struct wl_pointer *pointer,
uint32_t serial, struct wl_surface *surface, uint32_t serial, struct wl_surface *surface,
wl_fixed_t x, wl_fixed_t y) { wl_fixed_t x, wl_fixed_t y) {
fprintf(stderr, "pointer enter: %d, %d\n", x, y); // fprintf(stderr, "pointer enter: %d, %d\n", x, y);
} }
static void _wayland_pointer_leave(void *data, struct wl_pointer *pointer, static void _wayland_pointer_leave(void *data, struct wl_pointer *pointer,
uint32_t serial, uint32_t serial,
struct wl_surface *surface) { struct wl_surface *surface) {
fprintf(stderr, "pointer leave\n"); // fprintf(stderr, "pointer leave\n");
} }
static void _wayland_pointer_button(void *data, struct wl_pointer *pointer, static void _wayland_pointer_button(void *data, struct wl_pointer *pointer,
uint32_t serial, uint32_t time, uint32_t serial, uint32_t time,
@ -1046,6 +1052,11 @@ static void _opengl_gfx_present_wayland(_opengl_gfx_context_wayland *cx) {
(float)_gfx_context.frame_width, (float)_gfx_context.frame_width,
(float)_gfx_context.frame_height, (float)_gfx_context.frame_height,
}, },
.font_size =
{
(float)_FONT_WIDTH,
(float)_FONT_HEIGHT,
},
}; };
gfx_update_buffer(&_gfx_context, 3, &gpu_uniform_params, gfx_update_buffer(&_gfx_context, 3, &gpu_uniform_params,

View File

@ -3,10 +3,10 @@
#ifndef ED_HT_INCLUDED #ifndef ED_HT_INCLUDED
#define ED_HT_INCLUDED #define ED_HT_INCLUDED
#include <stdlib.h>
#include <stddef.h>
#include <stdbool.h> #include <stdbool.h>
#include <stddef.h>
#include <stdint.h> #include <stdint.h>
#include <stdlib.h>
#include <string.h> #include <string.h>
#include "string.h" #include "string.h"
@ -33,7 +33,7 @@ typedef uint64_t ht_hash_t;
static ht_hash_t ht_hash(string key, uint64_t mod) { static ht_hash_t ht_hash(string key, uint64_t mod) {
ht_hash_t hash = FNV_OFFSET; ht_hash_t hash = FNV_OFFSET;
for (size_t i=0; i<key.len; ++i) { for (size_t i = 0; i < key.len; ++i) {
hash *= FNV_PRIME; hash *= FNV_PRIME;
hash ^= key.data[i]; hash ^= key.data[i];
} }
@ -42,13 +42,14 @@ static ht_hash_t ht_hash(string key, uint64_t mod) {
} }
ed_ht ht_create(size_t max_entries, size_t value_size) { ed_ht ht_create(size_t max_entries, size_t value_size) {
ed_ht_slot *key_slots = (ed_ht_slot *)malloc(max_entries * sizeof(ed_ht_slot)); ed_ht_slot *key_slots =
(ed_ht_slot *)malloc(max_entries * sizeof(ed_ht_slot));
void *value_slots = malloc(max_entries * value_size); void *value_slots = malloc(max_entries * value_size);
memset(key_slots, 0, max_entries * sizeof(ed_ht_slot)); memset(key_slots, 0, max_entries * sizeof(ed_ht_slot));
memset(value_slots, 0, max_entries * value_size); memset(value_slots, 0, max_entries * value_size);
return (ed_ht) { return (ed_ht){
.key_slots = key_slots, .key_slots = key_slots,
.value_slots = value_slots, .value_slots = value_slots,
.value_size = value_size, .value_size = value_size,
@ -59,14 +60,15 @@ ed_ht ht_create(size_t max_entries, size_t value_size) {
bool ht_set(ed_ht *ht, string key, void *value) { bool ht_set(ed_ht *ht, string key, void *value) {
ht_hash_t hash = ht_hash(key, ht->capacity); ht_hash_t hash = ht_hash(key, ht->capacity);
for (size_t i=hash; i<ht->capacity; ++i) { for (size_t i = hash; i < ht->capacity; ++i) {
if (ht->key_slots[i].key.data == NULL || string_eq(ht->key_slots[i].key, key)) { if (ht->key_slots[i].key.data == NULL ||
string_eq(ht->key_slots[i].key, key)) {
if (ht->key_slots[i].key.data != NULL) { if (ht->key_slots[i].key.data != NULL) {
free(ht->key_slots[i].key.data); free(ht->key_slots[i].key.data);
} }
ht->key_slots[i].key = string_copy(key); ht->key_slots[i].key = string_copy(key);
memcpy(ht->value_slots+i*ht->value_size, value, ht->value_size); memcpy(ht->value_slots + i * ht->value_size, value, ht->value_size);
return true; return true;
} }
} }
@ -75,7 +77,7 @@ bool ht_set(ed_ht *ht, string key, void *value) {
} }
void *ht_get_slot(ed_ht *ht, size_t slot) { void *ht_get_slot(ed_ht *ht, size_t slot) {
void *value = ht->value_slots+slot*ht->value_size; void *value = ht->value_slots + slot * ht->value_size;
if (slot >= ht->capacity || !value) if (slot >= ht->capacity || !value)
return NULL; return NULL;
@ -86,7 +88,7 @@ void *ht_get_slot(ed_ht *ht, size_t slot) {
void *ht_get(ed_ht *ht, string key) { void *ht_get(ed_ht *ht, string key) {
ht_hash_t hash = ht_hash(key, ht->capacity); ht_hash_t hash = ht_hash(key, ht->capacity);
for (size_t i=hash; i<ht->capacity; ++i) { for (size_t i = hash; i < ht->capacity; ++i) {
if (ht->key_slots[i].key.data == NULL) { if (ht->key_slots[i].key.data == NULL) {
return NULL; return NULL;
} }
@ -102,13 +104,15 @@ void *ht_get(ed_ht *ht, string key) {
void ht_remove(ed_ht *ht, string key) { void ht_remove(ed_ht *ht, string key) {
ht_hash_t hash = ht_hash(key, ht->capacity); ht_hash_t hash = ht_hash(key, ht->capacity);
for (size_t i=hash; i<ht->capacity; ++i) { for (size_t i = hash; i < ht->capacity; ++i) {
if (ht->key_slots[i].key.data == NULL) { if (ht->key_slots[i].key.data == NULL) {
return; return;
} }
if (string_eq(ht->key_slots[i].key, key)) { if (string_eq(ht->key_slots[i].key, key)) {
ht->key_slots[i] = (ed_ht_slot) { 0 }; free(ht->key_slots[i].key.data);
ht->key_slots[i] = (ed_ht_slot){0};
return;
} }
} }
} }

View File

@ -15,13 +15,14 @@
#define ED_BUFFER_IMPLEMENTATION #define ED_BUFFER_IMPLEMENTATION
#define ED_FILE_IO_IMPLEMENTATION #define ED_FILE_IO_IMPLEMENTATION
#define CHAT_SLACK_IMPLEMENTATION #define CHAT_SLACK_IMPLEMENTATION
#include "ui.h"
#include "ed_array.h" #include "ed_array.h"
#include "file_io.h" #include "file_io.h"
#include "gfx.h" #include "gfx.h"
#include "ht.h" #include "ht.h"
#include "slack_api.h" #include "slack_api.h"
#include "string.h" #include "string.h"
#include "ui.h"
// static Arena default_arena = {0}; // static Arena default_arena = {0};
// static Arena temporary_arena = {0}; // static Arena temporary_arena = {0};
@ -45,14 +46,15 @@ static struct {
slack_user_info slack_user_info; slack_user_info slack_user_info;
array(slack_channel) slack_channels; array(slack_channel) slack_channels;
// TODO: store these for all channels ed_ht slack_messages;
array(slack_message) slack_messages; ed_ht slack_users;
size_t selected_channel_idx; // strictly a weak reference to a channel id
string selected_channel_idx;
} state; } state;
void init(_gfx_frame_func frame_func) { void init(_gfx_frame_func frame_func) {
state.gfx_cx = gfx_init_context(frame_func, 640, 480); state.gfx_cx = gfx_init_context(frame_func, 1280, 720);
state.ui_cx = ui_init_context(); state.ui_cx = ui_init_context();
// TODO: grab default font from the system // TODO: grab default font from the system
@ -200,7 +202,19 @@ void ed_frame(int mouse_x, int mouse_y, bool mouse_left_down,
for (int i = 0; i < state.slack_channels.size; ++i) { for (int i = 0; i < state.slack_channels.size; ++i) {
if (ui_button(&state.ui_cx, state.slack_channels.data[i].name) if (ui_button(&state.ui_cx, state.slack_channels.data[i].name)
.clicked) { .clicked) {
state.selected_channel_idx = i; state.selected_channel_idx =
state.slack_channels.data[i].id;
// FIXME: DO NOT DO THIS ON THE UI THREAD!
if (!ht_get(&state.slack_messages,
state.selected_channel_idx)) {
array(slack_message) messages = slack_api_message_list(
&state.slack_client, state.selected_channel_idx);
ht_set(&state.slack_messages,
state.selected_channel_idx, &messages);
}
} }
} }
@ -245,9 +259,41 @@ void ed_frame(int mouse_x, int mouse_y, bool mouse_left_down,
} }
ui_pop_parent(&state.ui_cx); ui_pop_parent(&state.ui_cx);
for (int i = 0; i < state.slack_messages.size; ++i) { ui_element(&state.ui_cx, _String("channel contents"), UI_AXIS_VERTICAL,
ui_button(&state.ui_cx, state.slack_messages.data[i].normal.text); ui_make_size(ui_fill, ui_fill), UI_FLAG_DRAW_BACKGROUND);
ui_push_parent(&state.ui_cx);
{
if (state.selected_channel_idx.data != NULL) {
array(slack_message) *messages =
ht_get(&state.slack_messages, state.selected_channel_idx);
if (messages) {
uint8_t text_buf[1024] = {0};
for (int i = messages->size - 1; i >= 0; --i) {
string message_text = messages->data[i].normal.text;
slack_user *user = ht_get(
&state.slack_users, messages->data[i].normal.user);
if (user) {
snprintf(text_buf, 1024, "%.*s: %.*s",
(int)user->real_name.len, user->name.data,
(int)message_text.len, message_text.data);
} else {
snprintf(text_buf, 1024, "Unknown: %.*s",
(int)message_text.len, message_text.data);
}
ui_button(&state.ui_cx, _CString_To_String(text_buf));
}
}
} else {
ui_element(&state.ui_cx, _String("no channel selected"),
UI_AXIS_VERTICAL, ui_make_size(ui_fill, ui_fit_text),
UI_FLAG_DRAW_BACKGROUND | UI_FLAG_DRAW_TEXT |
UI_FLAG_CENTERED_TEXT);
}
} }
ui_pop_parent(&state.ui_cx);
} }
ui_pop_parent(&state.ui_cx); ui_pop_parent(&state.ui_cx);
@ -283,9 +329,19 @@ int main(int argc, char *argv[]) {
state.slack_user_info = slack_api_auth_test(&state.slack_client); state.slack_user_info = slack_api_auth_test(&state.slack_client);
state.slack_channels = state.slack_channels =
slack_api_channel_list(&state.slack_client, state.slack_user_info); slack_api_channel_list(&state.slack_client, state.slack_user_info);
state.slack_messages = slack_api_message_list( state.selected_channel_idx = (string){0};
&state.slack_client, state.slack_channels.data[0].id);
state.selected_channel_idx = 0; // state.slack_messages = slack_api_message_list(
// &state.slack_client, state.slack_channels.data[0].id);
state.slack_messages = ht_create(1000, sizeof(array(slack_message)));
state.slack_users = ht_create(1000, sizeof(slack_user));
// array(slack_user) users =
// slack_api_user_list(&state.slack_client,
// state.slack_user_info.team_id);
// for (int i = 0; i < users.size; ++i) {
// ht_set(&state.slack_users, users.data[i].id, &users.data[i]);
// }
init(ed_frame); init(ed_frame);

View File

@ -52,11 +52,20 @@ typedef struct {
} slack_message; } slack_message;
arrayTemplate(slack_message); arrayTemplate(slack_message);
// TODO: merge with `slack_user_info`
typedef struct {
string id;
string name;
string real_name;
} slack_user;
arrayTemplate(slack_user);
slack_user_info slack_api_auth_test(slack_client *client); slack_user_info slack_api_auth_test(slack_client *client);
array(slack_channel) array(slack_channel)
slack_api_channel_list(slack_client *client, slack_user_info user); slack_api_channel_list(slack_client *client, slack_user_info user);
array(slack_message) array(slack_message)
slack_api_message_list(slack_client *client, string channel_id); slack_api_message_list(slack_client *client, string channel_id);
array(slack_user) slack_api_user_list(slack_client *client, string team_id);
slack_client slack_init_client(); slack_client slack_init_client();
@ -208,9 +217,10 @@ array(slack_channel)
if (json == NULL) { if (json == NULL) {
const char *error_ptr = cJSON_GetErrorPtr(); const char *error_ptr = cJSON_GetErrorPtr();
if (error_ptr != NULL) { if (error_ptr != NULL) {
fprintf(stderr, fprintf(
"SLACK CLIENT: Failed to parse /auth.test: %s\n", stderr,
error_ptr); "SLACK CLIENT: Failed to parse /conversations.list: %s\n",
error_ptr);
// FIXME: don't panic here // FIXME: don't panic here
exit(1); exit(1);
@ -253,11 +263,13 @@ array(slack_channel)
const cJSON *error = const cJSON *error =
cJSON_GetObjectItemCaseSensitive(json, "error"); cJSON_GetObjectItemCaseSensitive(json, "error");
if (cJSON_IsString(error)) { if (cJSON_IsString(error)) {
fprintf(stderr, "SLACK CLIENT: /auth.test failed: %s\n", fprintf(stderr,
"SLACK CLIENT: /conversations.list failed: %s\n",
error->valuestring); error->valuestring);
} else { } else {
fprintf(stderr, "SLACK CLIENT: /auth.test failed: failed " fprintf(stderr,
"to parse 'error' field\n"); "SLACK CLIENT: /conversations.list failed: failed "
"to parse 'error' field\n");
} }
} }
} }
@ -321,7 +333,8 @@ array(slack_message)
const char *error_ptr = cJSON_GetErrorPtr(); const char *error_ptr = cJSON_GetErrorPtr();
if (error_ptr != NULL) { if (error_ptr != NULL) {
fprintf(stderr, fprintf(stderr,
"SLACK CLIENT: Failed to parse /auth.test: %s\n", "SLACK CLIENT: Failed to parse /conversations.history: "
"%s\n",
error_ptr); error_ptr);
// FIXME: don't panic here // FIXME: don't panic here
@ -373,11 +386,14 @@ array(slack_message)
const cJSON *error = const cJSON *error =
cJSON_GetObjectItemCaseSensitive(json, "error"); cJSON_GetObjectItemCaseSensitive(json, "error");
if (cJSON_IsString(error)) { if (cJSON_IsString(error)) {
fprintf(stderr, "SLACK CLIENT: /auth.test failed: %s\n", fprintf(stderr,
"SLACK CLIENT: /conversations.history failed: %s\n",
error->valuestring); error->valuestring);
} else { } else {
fprintf(stderr, "SLACK CLIENT: /auth.test failed: failed " fprintf(
"to parse 'error' field\n"); stderr,
"SLACK CLIENT: /conversations.history failed: failed "
"to parse 'error' field\n");
} }
} }
} }
@ -390,6 +406,114 @@ array(slack_message)
return (array(slack_message)){0}; return (array(slack_message)){0};
} }
array(slack_user) slack_api_user_list(slack_client *client, string team_id) {
if (client->curl) {
struct curl_slist *headers = NULL;
headers = curl_slist_append(
headers, "Content-Type: application/x-www-form-urlencoded");
headers = curl_slist_append(headers, client->cookie.data);
curl_easy_setopt(client->curl, CURLOPT_URL,
"https://slack.com/api/users.list");
curl_easy_setopt(client->curl, CURLOPT_HTTPAUTH, (long)CURLAUTH_BEARER);
curl_easy_setopt(client->curl, CURLOPT_XOAUTH2_BEARER,
client->token.data);
curl_easy_setopt(client->curl, CURLOPT_USERAGENT,
"Mozilla/5.0 (Macintosh; Intel Mac OS X 10.15; "
"rv:76.0) Gecko/20100101 Firefox/76.0");
curl_easy_setopt(client->curl, CURLOPT_HTTPHEADER, headers);
curl_easy_setopt(client->curl, CURLOPT_POST, 1L);
char *fields_buffer[2048];
snprintf(fields_buffer, 2048, "team_id=%.*s", (int)team_id.len,
team_id.data);
fprintf(stderr, "fields_buffer: %s\n", fields_buffer);
curl_easy_setopt(client->curl, CURLOPT_POSTFIELDS, fields_buffer);
curl_easy_setopt(client->curl, CURLOPT_FOLLOWLOCATION, 1L);
// TODO: don't allocate this on every request
array(uint8_t) chunk = newArray(uint8_t, 4096);
curl_easy_setopt(client->curl, CURLOPT_WRITEFUNCTION,
write_memory_callback);
curl_easy_setopt(client->curl, CURLOPT_WRITEDATA, (void *)&chunk);
CURLcode res = curl_easy_perform(client->curl);
if (res != CURLE_OK) {
fprintf(stderr, "curl_easy_perform() failed: %s\n",
curl_easy_strerror(res));
} else {
fprintf(stderr, "Got data\n%.*s\n", (int)chunk.size, chunk.data);
}
// parse json
cJSON *json = cJSON_ParseWithLength(chunk.data, chunk.size);
if (json == NULL) {
const char *error_ptr = cJSON_GetErrorPtr();
if (error_ptr != NULL) {
fprintf(stderr,
"SLACK CLIENT: Failed to parse /users.list: %s\n",
error_ptr);
// FIXME: don't panic here
exit(1);
}
}
{
const cJSON *ok = cJSON_GetObjectItemCaseSensitive(json, "ok");
if (cJSON_IsBool(ok) && ok->valueint == 1) {
array(slack_user) users = newArray(slack_user, 10);
fprintf(stderr, "SLACK CLIENT: /users.list succeeded!\n");
const cJSON *json_users =
cJSON_GetObjectItemCaseSensitive(json, "members");
const cJSON *json_user;
cJSON_ArrayForEach(json_user, json_users) {
const cJSON *user_id =
cJSON_GetObjectItemCaseSensitive(json_user, "id");
const cJSON *name =
cJSON_GetObjectItemCaseSensitive(json_user, "name");
const cJSON *real_name = cJSON_GetObjectItemCaseSensitive(
json_user, "real_name");
slack_user user = {
.id = string_copy_cstring(user_id->valuestring),
.name = string_copy_cstring(name->valuestring),
.real_name =
string_copy_cstring(real_name->valuestring),
};
pushArray(slack_user, &users, user);
}
return users;
} else {
const cJSON *error =
cJSON_GetObjectItemCaseSensitive(json, "error");
if (cJSON_IsString(error)) {
fprintf(stderr, "SLACK CLIENT: /users.list failed: %s\n",
error->valuestring);
} else {
fprintf(stderr, "SLACK CLIENT: /users.list failed: failed "
"to parse 'error' field\n");
}
}
}
cJSON_Delete(json);
free(chunk.data);
}
// TODO: create some slack api error type
return (array(slack_user)){0};
}
slack_client slack_init_client(string token, string cookie) { slack_client slack_init_client(string token, string cookie) {
CURL *curl = curl_easy_init(); CURL *curl = curl_easy_init();

View File

@ -3,11 +3,12 @@
#ifndef ED_UI_INCLUDED #ifndef ED_UI_INCLUDED
#define ED_UI_INCLUDED #define ED_UI_INCLUDED
#include "string.h"
#define MAX_UI_ELEMENTS 8192 #define MAX_UI_ELEMENTS 8192
// TODO: replace this with functions // TODO: replace this with functions
#define _FONT_WIDTH 12
#define _FONT_HEIGHT 24 #define _FONT_HEIGHT 24
#define _FONT_WIDTH _FONT_HEIGHT / 2
#define _elm(index) (cx->frame_elements.data + index) #define _elm(index) (cx->frame_elements.data + index)
#define _flags(index, flgs) ((_elm(index)->flags & (flgs)) == (flgs)) #define _flags(index, flgs) ((_elm(index)->flags & (flgs)) == (flgs))
@ -195,10 +196,10 @@ size_t ui_element(ui_context *cx, string label, ui_axis axis,
ui_semantic_size size[2], ui_flags flags) { ui_semantic_size size[2], ui_flags flags) {
ui_element_frame_data frame_data = (ui_element_frame_data){ ui_element_frame_data frame_data = (ui_element_frame_data){
.index = cx->frame_elements.size, .index = cx->frame_elements.size,
// TODO: don't just set this to label, because then elements // TODO: don't just set `key` to label, because then elements
// with the same label can't be created together // with the same label can't be created together
.key = label, .key = string_copy(label),
.label = label, .label = string_copy(label),
.first = -1, .first = -1,
.last = -1, .last = -1,
.next = -1, .next = -1,
@ -223,7 +224,7 @@ size_t ui_element(ui_context *cx, string label, ui_axis axis,
} else { } else {
bool did_insert = ht_set(&cx->cached_elements, label, bool did_insert = ht_set(&cx->cached_elements, label,
&(ui_element_cache_data){ &(ui_element_cache_data){
.label = label, .label = string_copy(label),
.size = {0}, .size = {0},
.last_instantiated_index = cx->frame_index, .last_instantiated_index = cx->frame_index,
}); });
@ -560,6 +561,7 @@ void ui_prune(ui_context *cx) {
// %zu, frame index: %zu\n", (int)key.len, key.data, // %zu, frame index: %zu\n", (int)key.len, key.data,
// cached->last_instantiated_index, cx->frame_index); // cached->last_instantiated_index, cx->frame_index);
free(cached->label.data);
ht_remove(&cx->cached_elements, key); ht_remove(&cx->cached_elements, key);
} }
} }