2022-01-05 18:53:44 +01:00
local st = require " util.stanza " ;
local http = require " net.http " ;
2022-01-05 19:17:40 +01:00
local gettime = require ' socket ' . gettime ;
2022-01-05 18:53:44 +01:00
local async = require " util.async " ;
local b64 = require " util.encodings " . base64.encode ;
local jid_split = require " util.jid " . split ;
local json = require " util.json " ;
local uh = require " util.http " ;
module : add_feature ( " vcard-temp " ) ;
2022-01-05 19:17:40 +01:00
local CACHE_EXPIRY = 3600 ;
local cache_user = { } ;
2022-01-05 18:53:44 +01:00
local peertube_url = assert ( module : get_option_string ( " peertubelivechat_vcard_peertube_url " , nil ) , " 'peertubelivechat_vcard_peertube_url' is a required option " ) ;
if peertube_url : sub ( - 1 , - 1 ) == " / " then peertube_url = peertube_url : sub ( 1 , - 2 ) ; end
module : hook ( " iq-get/bare/vcard-temp:vCard " , function ( event )
local origin , stanza = event.origin , event.stanza ;
local who = jid_split ( stanza.attr . to ) or origin.username
module : log ( " debug " , " vCard request for %s " , who ) ;
2022-01-05 19:17:40 +01:00
local from_cache = cache_user [ who ] ;
if from_cache then
if from_cache [ " last_fetch_time " ] and from_cache [ " last_fetch_time " ] + CACHE_EXPIRY < gettime ( ) then
module : log ( " debug " , " vCard result for %s was in cache but is expired. " , who ) ;
cache_user [ who ] = nil
else
module : log ( " debug " , " vCard result for %s is in cache. " , who ) ;
if ( from_cache [ ' vcard ' ] ) then
origin.send ( st.reply ( stanza ) : add_child ( from_cache [ " vcard " ] ) ) ;
else
origin.send ( st.error_reply ( stanza , " cancel " , " item-not-found " ) ) ;
end
return true ;
end
else
module : log ( " debug " , " vCard result for %s is not in cache. " , who ) ;
end
2022-01-05 18:53:44 +01:00
local wait , done = async.waiter ( ) ;
local url = peertube_url .. ' /api/v1/accounts/ ' .. uh.urlencode ( who ) ;
module : log ( " debug " , " Calling Peertube API: %s " , url ) ;
local ret ;
http.request ( url , { accept = " application/json " } , function ( body , code )
if math.floor ( code / 100 ) == 2 then
local parsed , parse_err = json.decode ( body ) ;
if not parsed then
module : log ( " debug " , " Got invalid JSON from %s: %s " , url , parse_err ) ;
else
ret = parsed ;
end
else
module : log ( " debug " , " Rejected by API: " , body ) ;
end
done ( ) ;
end )
wait ( ) ;
if not ret then
module : log ( " debug " , " Peertube user not found, no vCard for %s " , who ) ;
origin.send ( st.error_reply ( stanza , " cancel " , " item-not-found " ) ) ;
2022-01-05 19:17:40 +01:00
cache_user [ who ] = { last_fetch_time = gettime ( ) } ;
2022-01-05 18:53:44 +01:00
return true ;
end
2022-01-05 20:13:59 +01:00
2022-01-05 18:53:44 +01:00
local vcard_temp = st.stanza ( " vCard " , { xmlns = " vcard-temp " } ) ;
2022-01-05 19:23:20 +01:00
vcard_temp : text_tag ( " FN " , ret.displayName ) ;
2022-01-05 18:53:44 +01:00
vcard_temp : text_tag ( " NICKNAME " , ret.displayName ) ;
2022-01-05 20:11:22 +01:00
vcard_temp : text_tag ( " URL " , ret.url ) ;
2022-01-05 20:13:59 +01:00
2022-01-05 18:53:44 +01:00
if ret.avatar and ret.avatar . path then
module : log ( " debug " , " Downloading user avatar on %s " , peertube_url .. ret.avatar . path ) ;
local waitAvatar , doneAvatar = async.waiter ( ) ;
http.request ( peertube_url .. ret.avatar . path , { } , function ( body , code , response )
if math.floor ( code / 100 ) == 2 then
module : log ( " debug " , " Avatar found for %s " , who ) ;
2022-01-06 03:29:52 +01:00
vcard_temp : tag ( " PHOTO " ) ;
2022-01-05 20:11:22 +01:00
if ( response and response.headers and response.headers [ " content-type " ] ) then
module : log ( " debug " , " Avatar content-type: %s " , response.headers [ " content-type " ] ) ;
2022-01-06 03:29:52 +01:00
vcard_temp : text_tag ( " TYPE " , response.headers [ " content-type " ] ) ;
vcard_temp : text_tag ( " BINVAL " , b64 ( body ) ) ;
vcard_temp : up ( ) ;
2022-01-05 20:11:22 +01:00
else
module : log ( " debug " , " Avatar has no content-type. " ) ;
2022-01-05 18:53:44 +01:00
end
else
module : log ( " debug " , " Cant load avatar: " , body ) ;
end
doneAvatar ( ) ;
end )
waitAvatar ( ) ;
end
origin.send ( st.reply ( stanza ) : add_child ( vcard_temp ) ) ;
2022-01-05 19:17:40 +01:00
cache_user [ who ] = { last_fetch_time = gettime ( ) , vcard = vcard_temp } ;
2022-01-05 18:53:44 +01:00
return true ;
2022-01-05 20:13:59 +01:00
end ) ;