dbus: add client support

Signed-off-by: Julien Danjou <julien@danjou.info>
This commit is contained in:
Julien Danjou 2008-07-08 10:54:57 +02:00
parent 1d64f8231c
commit 3de2bf3918
4 changed files with 149 additions and 2 deletions

View File

@ -1122,7 +1122,7 @@ luaA_client_unmanage(lua_State *L)
* \param L The Lua VM state. * \param L The Lua VM state.
* \return The number of elements pushed on stack. * \return The number of elements pushed on stack.
*/ */
static int int
luaA_client_newindex(lua_State *L) luaA_client_newindex(lua_State *L)
{ {
size_t len; size_t len;

View File

@ -46,6 +46,8 @@ void client_setfloating(client_t *, bool);
char * client_markup_parse(client_t *, const char *, ssize_t); char * client_markup_parse(client_t *, const char *, ssize_t);
void client_setborder(client_t *, int); void client_setborder(client_t *, int);
int luaA_client_newindex(lua_State *);
int luaA_client_userdata_new(lua_State *, client_t *); int luaA_client_userdata_new(lua_State *, client_t *);
DO_SLIST(client_t, client, p_delete) DO_SLIST(client_t, client, p_delete)

View File

@ -11,8 +11,10 @@ border_width
bottom bottom
bottomleft bottomleft
bottomright bottomright
byname
center center
class class
client
color color
coords coords
fg fg

145
dbus.c
View File

@ -27,6 +27,7 @@
#include "dbus.h" #include "dbus.h"
#include "widget.h" #include "widget.h"
#include "client.h"
extern awesome_t globalconf; extern awesome_t globalconf;
@ -34,11 +35,129 @@ static DBusError err;
static DBusConnection *dbus_connection = NULL; static DBusConnection *dbus_connection = NULL;
ev_io dbusio = { .fd = -1 }; ev_io dbusio = { .fd = -1 };
/** Check a dbus object path format and its number of element.
* \param path The path.
* \param nelem The number of element it should have.
* \return true if the path is ok, false otherwise.
*/
static bool
a_dbus_path_check(char **path, int nelem)
{
int i;
for(i = 0; path[i]; i++);
if(i != nelem)
return false;
return (!a_strcmp(path[0], "org") && !a_strcmp(path[1], "awesome"));
}
static void
a_dbus_process_request_client(DBusMessage *msg)
{
int i = 2;
DBusMessageIter iter;
const char *m;
char **path, *arg;
client_t *c;
bool b;
uint64_t i64;
double d;
if(!dbus_message_get_path_decomposed(msg, &path))
return;
/* path is:
* /org/awesome/clients/bywhat/id */
if(!a_dbus_path_check(path, 5)
|| a_strcmp(path[2], "clients"))
goto bailout;
/* handle by- */
switch(a_tokenize(path[3], -1))
{
case A_TK_BYNAME:
for(c = globalconf.clients; c; c = c->next)
if(!a_strcmp(c->name, path[4]))
break;
/* not found, return */
if(!c)
goto bailout;
printf("found\n");
break;
default:
return;
}
if(!dbus_message_iter_init(msg, &iter))
{
printf("bad init\n");
dbus_error_free(&err);
return;
}
/* clear stack */
lua_settop(globalconf.L, 0);
/* push index function */
lua_pushcfunction(globalconf.L, luaA_client_newindex);
/* push the client */
luaA_client_userdata_new(globalconf.L, c);
/* get the method name */
m = dbus_message_get_member(msg);
/* push the method name */
lua_pushstring(globalconf.L, m);
do
{
switch(dbus_message_iter_get_arg_type(&iter))
{
case DBUS_TYPE_STRING:
dbus_message_iter_get_basic(&iter, &arg);
lua_pushstring(globalconf.L, arg);
break;
case DBUS_TYPE_BOOLEAN:
dbus_message_iter_get_basic(&iter, &b);
lua_pushboolean(globalconf.L, b);
break;
case DBUS_TYPE_INT16:
case DBUS_TYPE_UINT16:
case DBUS_TYPE_INT32:
case DBUS_TYPE_UINT32:
case DBUS_TYPE_INT64:
case DBUS_TYPE_UINT64:
dbus_message_iter_get_basic(&iter, &i64);
lua_pushnumber(globalconf.L, i64);
break;
case DBUS_TYPE_DOUBLE:
dbus_message_iter_get_basic(&iter, &d);
lua_pushnumber(globalconf.L, d);
break;
default:
break;
}
i++;
}
while(dbus_message_iter_next(&iter));
lua_pcall(globalconf.L, i, 0, 0);
printf("pcall\n");
bailout:
for(i = 0; path[i]; i++)
p_delete(&path[i]);
p_delete(&path);
}
static void static void
a_dbus_process_requests(EV_P_ ev_io *w, int revents) a_dbus_process_requests(EV_P_ ev_io *w, int revents)
{ {
DBusMessage *msg; DBusMessage *msg;
int nmsg = 0; int nmsg = 0;
const char *n;
ssize_t len;
if(!dbus_connection && !a_dbus_init()) if(!dbus_connection && !a_dbus_init())
return; return;
@ -50,13 +169,37 @@ a_dbus_process_requests(EV_P_ ev_io *w, int revents)
if(!(msg = dbus_connection_pop_message(dbus_connection))) if(!(msg = dbus_connection_pop_message(dbus_connection)))
break; break;
if(dbus_message_is_signal(msg, DBUS_INTERFACE_LOCAL, "Disconnected")) if(dbus_message_is_signal(msg, DBUS_INTERFACE_LOCAL, "Disconnected"))
{ {
a_dbus_cleanup(); a_dbus_cleanup();
dbus_message_unref(msg); dbus_message_unref(msg);
return; return;
} }
else
{
/* get the interface */
n = dbus_message_get_interface(msg);
len = a_strlen(n);
/* check destination: at least has sizeof(org.awesome.)
* and match org.awesome */
if(len > 12 && !a_strncmp(n, "org.awesome", 11))
{
/* n is now after at the end of 'org.awesome.' */
n += 12;
/* we assume that after the dot there's a class name, like
* widget, client, etc */
switch(a_tokenize(n, -1))
{
case A_TK_CLIENT:
a_dbus_process_request_client(msg);
break;
default:
break;
}
}
}
dbus_message_unref(msg); dbus_message_unref(msg);