1
0
mirror of https://github.com/fazo96/telecommander.git synced 2025-01-10 11:34:20 +01:00
telecommander/telecommander.js

384 lines
13 KiB
JavaScript
Raw Normal View History

2015-08-29 02:38:59 +02:00
#!/usr/bin/env node
2015-08-31 18:37:32 +02:00
var data = {} // Hold all global data
2015-08-29 02:38:59 +02:00
var os = require('os')
var fs = require('fs')
2015-08-30 01:49:03 +02:00
var moment = require('moment')
2015-08-29 02:38:59 +02:00
var blessed = require('blessed')
2015-08-31 14:04:32 +02:00
var path = require('path')
data.cfgDir = path.join(process.env.XDG_CONFIG_HOME || (path.join(process.env.HOME || process.env.USERPROFILE, '/.config/')), 'telecommander/')
2015-08-31 18:37:32 +02:00
process.env.LOGGER_FILE = process.env.LOGGER_FILE || "/tmp/telecommander"
data.keyFile = path.join(data.cfgDir,'key')
data.userFile = path.join(data.cfgDir,'user_data.json')
data.telegramLink = require('telegram.link')()
// Load modules
require('./lib/cli.js')(data) // Parse command line args
require('./lib/util.js')(data) // Load utils
require('./lib/ui.js')(data) // Load ui
2015-08-31 14:04:32 +02:00
2015-08-30 16:40:01 +02:00
/* IF YOU FORK THE APP PLEASE CHANGE THE ID
* AND HASH IN THE APP OBJECT! THEY IDENTIFY
* THE APPLICATION CREATOR AND YOU CAN
* OBTAIN YOURS FROM http://my.telegram.org
*/
2015-08-31 14:04:32 +02:00
data.app = {
2015-08-30 16:40:01 +02:00
id: '42419',
hash: '90a3c2cdbf9b391d9ed72c0639dc0786',
version: require('./package.json').version,
lang: 'en',
deviceModel: os.type(),
systemVersion: os.platform()+'/'+os.release()
}
2015-08-31 14:04:32 +02:00
// Logger
2015-08-29 02:38:59 +02:00
var getLogger = require('get-log')
getLogger.PROJECT_NAME = 'telecommander'
2015-08-31 14:04:32 +02:00
data.logger = getLogger('main')
2015-08-29 02:38:59 +02:00
2015-08-31 14:04:32 +02:00
data.authKey // our authorization key to access telegram
data.connected = false // keep track of wether we are good to go and logged in
2015-08-29 02:38:59 +02:00
2015-08-31 00:19:30 +02:00
// Write something in the Status box
2015-08-31 14:04:32 +02:00
data.log = function(){
2015-08-31 00:19:30 +02:00
args = Array.prototype.slice.call(arguments)
var msg = args.join(' ')
2015-08-31 14:04:32 +02:00
data.getMsgBox(data.statusWindow).add(msg)
data.logger.info(msg)
2015-08-31 00:19:30 +02:00
}
2015-08-31 14:04:32 +02:00
data.command = function(cmd){
cmdl = cmd.split(' ')
cmdname = cmdl[0]
2015-08-29 02:38:59 +02:00
2015-08-29 13:51:14 +02:00
if(cmdname === 'phone'){ // So the user can provide his phone numbah
2015-08-31 18:37:32 +02:00
if(data.connected){
2015-08-29 13:51:14 +02:00
return log("Silly user, you're already connected! We don't need that phone number")
}
2015-08-29 02:38:59 +02:00
2015-08-29 13:51:14 +02:00
} else if(cmdname === 'code'){ // So the user can provide his phone code
2015-08-31 18:37:32 +02:00
if(data.connected){
return data.log("Silly user, you're already connected! We don't need that phone code")
2015-08-29 13:51:14 +02:00
}
2015-08-31 00:19:30 +02:00
} else {
2015-08-31 18:37:32 +02:00
data.log('Command not found.')
2015-08-29 02:38:59 +02:00
}
}
2015-08-29 13:51:14 +02:00
// Send a message
2015-08-31 14:04:32 +02:00
data.sendMsg = function(name,str){
if(!data.connected){
2015-08-29 13:51:14 +02:00
return log('Error: not ready to send messages')
}
2015-08-31 14:04:32 +02:00
var obj = data.nameToObj(name)
var peer = data.idToPeer(obj.id,obj.title?'group':'user')
var randid = parseInt(Math.random() * 999999999)
// Fix bug in telegram.link that doesn't send strings with accented letters
str = str.replace('è',"e'").replace('ù',"u'").replace('à',"a'").replace('ò',"o'").replace('ì',"i'")
//data.log('Sending Message to:',peer.toPrintable())
2015-08-31 14:04:32 +02:00
data.client.messages.sendMessage(peer,str,randid,function(sent){
data.log('Sent message:','"'+str+'"','to:',data.selectedWindow+':',sent.toPrintable())
2015-08-29 13:51:14 +02:00
})
}
data.onPhoneCode = function(something,s){
if(s === null){ // User cancelled
process.exit(0)
}
var cmdl = s.split(' ')
code = cmdl[0]
name = cmdl[1]
lastname = cmdl[2]
if(((!name || !lastname) && !data.user.registered) || !code)
return log('insufficient arguments:',cmd) // TODO: handle this better!
var cb = function(result){
data.user.id = ''+result.user.id
data.user.phone = result.user.phone
data.user.phoneCodeHash = result.phone_code_hash
data.user.username = result.user.username
data.user.first_name = result.user.first_name
data.user.last_name = result.user.last_name
// Done, write user data and key to disk
try {
fs.mkdirSync(data.cfgDir,'0770')
} catch (e) {
if(e.code != 'EEXIST'){
console.error("FATAL: couldn't create configuration directory",data.cfgDir,e)
process.exit(-1)
}
}
data.log('Writing Log In token and user data to',data.cfgDir)
fs.writeFile(data.cfgDir+'key',data.app.authKey,function(err){
if(err) data.log('FATAL: Could not write key to disk:',err)
})
fs.writeFile(data.cfgDir+'user_data.json',JSON.stringify(data.user),function(err){
if(err) data.log("FATAL: couldn't write user_data.json:",err)
})
data.whenReady()
}
// Log in finally
if(data.user.registered) data.client.auth.signIn(data.user.phone,data.user.phoneCodeHash,code,cb)
else data.client.auth.signUp(data.user.phone,data.user.phoneCodeHash,code,name,lastname,cb)
}
data.onPhoneNumber = function(something,s){
if(s === null){ // User cancelled
process.exit(0)
}
data.user.phone = s.trim()
var mindate = moment()
data.log('Checking your phone number with Telegram...')
data.client.auth.sendCode(data.user.phone,5,'en',function(result){
if(result.err_code){
return data.log('Errors:',result.error_code,result.error_message)
}
//data.log('Res:',JSON.stringify(result))
data.user.registered = result.phone_registered
data.user.phoneCodeHash = result.phone_code_hash
var msg
if(!data.user.registered){
msg = "Your number ("+data.user.phone+") is not registered.\nTelecommander will register your account with the Telegram service."
} else {
msg = "Your number ("+data.user.phone+") is already assigned to a Telegram account.\nTelecommander will log you in."
}
msg += "\nPress ESC to exit now, or enter to continue"
data.popup.display(msg,0,function(){
data.popup.hide()
data.promptBox.input('Your telegram code:','',data.onPhoneCode)
})
})
}
2015-08-29 02:38:59 +02:00
// Connects to telegram
2015-08-31 14:04:32 +02:00
data.connect = function(){
data.load('Connecting...')
2015-08-31 18:37:32 +02:00
data.client = data.telegramLink.createClient(data.app, data.dataCenter, function(){
2015-08-31 14:04:32 +02:00
if(!data.app.authKey){
2015-08-31 18:37:32 +02:00
data.log('Downloading Authorization Key...')
2015-08-31 14:04:32 +02:00
data.client.createAuthKey(function(auth){
data.app.authKey = auth.key.encrypt('password') // Will add security later, I promise
// Writes the new encrypted key to disk
data.loader.stop()
//data.log('Ready for phone number, use command: phone <number>')
data.promptBox.input('Phone number (international format):','+',data.onPhoneNumber)
2015-08-29 02:38:59 +02:00
})
} else {
2015-08-31 14:04:32 +02:00
data.whenReady()
2015-08-29 02:38:59 +02:00
}
})
2015-08-31 14:04:32 +02:00
data.client.once('dataCenter',function(dcs){
data.log('Datacenters:',dcs.toPrintable())
2015-08-29 02:38:59 +02:00
})
}
2015-08-29 13:51:14 +02:00
// Executed when connected and logged in
2015-08-31 14:04:32 +02:00
data.whenReady = function(){
data.load('Connected')
2015-08-31 14:04:32 +02:00
data.connected = true
data.downloadData()
data.chats.focus()
2015-08-29 02:38:59 +02:00
}
2015-08-29 13:51:14 +02:00
// Downloads stuff
2015-08-31 14:04:32 +02:00
data.downloadData = function(){
data.load('Downloading data')
2015-08-31 14:04:32 +02:00
data.client.contacts.getContacts('',function(cont){
//data.chats.clearItems()
//data.chats.add(data.statusWindow)
2015-08-31 14:04:32 +02:00
cont.users.list.forEach(data.addUser)
data.loader.stop()
2015-08-29 02:38:59 +02:00
})
2015-08-30 01:49:03 +02:00
2015-08-31 14:04:32 +02:00
data.client.messages.getDialogs(0,0,10,function(dialogs){
if(dialogs && dialogs.chats && dialogs.chats.list)
2015-08-31 14:04:32 +02:00
dialogs.chats.list.forEach(data.addGroup)
data.loader.stop()
2015-08-30 01:49:03 +02:00
})
2015-08-31 14:04:32 +02:00
data.client.updates.getState(function(astate){
data.updateState(astate)
data.log(data.state.unreadCount,'unread messages')
2015-08-31 14:35:08 +02:00
//data.log('Started receiving updates')
// Can't use registerOnUpdates because it's apparently broken
//client.registerOnUpdates(onUpdate)
2015-08-31 14:04:32 +02:00
setTimeout(data.downloadUpdates,1000)
2015-08-29 13:51:14 +02:00
})
}
2015-08-31 14:04:32 +02:00
data.downloadUpdates = function(){
data.client.updates.getDifference(data.state.pts,data.state.date,data.state.qts,function(res){
2015-08-30 16:40:01 +02:00
if(!res.instanceOf('api.type.updates.DifferenceEmpty')){
//data.log('Got Diff: ',res.toPrintable())
2015-08-30 16:40:01 +02:00
if(res.state){
2015-08-31 14:04:32 +02:00
data.updateState(res.state)
2015-08-30 16:40:01 +02:00
}
if(res.chats)
for(c in res.chats.list) data.addGroup(res.chats.list[c])
if(res.users)
for(c in res.users.list) data.addUser(res.users.list[c])
2015-08-30 16:40:01 +02:00
if(res.new_messages){
res.new_messages.list.forEach(function(msg){
data.appendMsg(msg,undefined,false,false)
2015-08-30 16:40:01 +02:00
})
}
data.rebuildChatList()
}
2015-08-31 14:04:32 +02:00
setTimeout(data.downloadUpdates,1000)
})
}
2015-08-30 20:29:02 +02:00
// Get message history with given name in the given box
// BROKEN, need to be rethinked
2015-08-31 14:04:32 +02:00
data.getMessages = function(name,box){
if(!data.connected){
return // data.log('Uh cant get messages cuz not connected.....')
2015-08-29 13:51:14 +02:00
}
if(data.downloadingMessages == true) return
2015-08-30 20:29:02 +02:00
//log('Name to obj:',name)
2015-08-31 14:04:32 +02:00
var obj = data.nameToObj(name)
2015-08-30 20:29:02 +02:00
if(!obj || !obj.id){
2015-08-31 18:37:32 +02:00
return //data.log("Can't get messages",obj,obj.id,obj.title)
2015-08-30 20:29:02 +02:00
}
var type = obj.title?'group':'user'
2015-08-31 14:04:32 +02:00
var peer = data.idToPeer(obj.id,type)
//box.add('Downloading message history for '+name)
2015-08-30 20:29:02 +02:00
if(!peer) return log('Could not find peer:',name)
data.downloadingMessages = true
var oldnlines = box.getLines().length
if(data.selectedWindow === name) data.load('Downloading history...')
data.client.messages.getHistory(peer,0,obj.oldest_message||0,10,function(res){
2015-08-30 01:49:03 +02:00
//log(res.toPrintable())
2015-08-31 03:03:28 +02:00
//log('Got history for: '+getName(peer.user_id||peer.chat_id,peer.chat_id?'group':'user'))
2015-08-30 16:40:01 +02:00
if(!res.messages){
return box.add(res.toPrintable())
}
2015-08-30 01:49:03 +02:00
res.messages.list.sort(function(msg1,msg2){
return msg1.date - msg2.date
})
res.messages.list.reverse()
2015-08-29 13:51:14 +02:00
res.messages.list.forEach(function(msg){
data.appendMsg(msg,undefined,false,true)
2015-08-29 13:51:14 +02:00
})
if(oldnlines == 0) box.setScrollPerc(100)
//box.add(obj.oldest_message)
data.loader.stop()
data.downloadingMessages = false
2015-08-29 02:38:59 +02:00
})
}
2015-08-31 14:04:32 +02:00
data.appendToUserBox = function(msg,context){
var goesto
if(context.messages.list.length > 0){
if(context.messages.list[0].to_id.chat_id){
// Group message
//data.log('Chose',data.getName(context.messages.list[0].to_id.chat_id,'group'))
2015-08-31 14:04:32 +02:00
goesto = data.getMsgBox(data.getName(context.messages.list[0].to_id.chat_id))
}
}
if(goesto === undefined){
2015-08-31 15:32:39 +02:00
if(context.users.list[0].user_id == data.user.id){
goesto = data.getMsgBox(data.getName(context.users.list[1].id,'user'))
} else{
2015-08-31 15:32:39 +02:00
goesto = data.getMsgBox(data.getName(context.users.list[0].id,'user'))
}
}
2015-08-31 14:04:32 +02:00
data.appendMsg(msg,goesto,true)
}
2015-08-30 01:49:03 +02:00
// Writes given telegram.link "message" object to given boxId
data.appendMsg = function(msg,toBoxId,bare,prepend){
var box,param,obj
if(toBoxId != undefined){
box = toBoxId
} else {
2015-08-30 20:29:02 +02:00
if(msg.to_id.chat_id != undefined){
// Is a group
2015-08-31 14:04:32 +02:00
param = data.getName(msg.to_id.chat_id,'group')
obj = data.groups[msg.to_id.chat_id]
2015-08-31 14:35:08 +02:00
} else if(msg.from_id === msg.to_id.user_id || msg.from_id != data.user.id){
2015-08-31 14:04:32 +02:00
param = data.getName(msg.from_id,'user')
obj = data.contacts[msg.from_id]
2015-08-31 14:35:08 +02:00
} else if(msg.to_id.user_id != undefined && msg.to_id.user_id != data.user.id) {
// don't forget dat .user_id! don't need it in from_id...
2015-08-31 14:04:32 +02:00
param = data.getName(msg.to_id.user_id,'user')
obj = data.contacts[msg.to_id.user_id]
}
// Increase unread count if necessary
if(data.selectedWindow != param || data.msgBox[param] === undefined){
if(!obj.toread) obj.toread = 1
else obj.toread++
}
// Update oldest and latest message reference
if(!obj.oldest_message || parseInt(obj.oldest_message) > parseInt(msg.id))
obj.oldest_message = parseInt(msg.id)
if(!obj.latest_message || parseInt(obj.latest_message) < parseInt(msg.id))
obj.latest_message = parseInt(msg.id)
2015-08-31 14:04:32 +02:00
box = data.getMsgBox(param)
}
if(bare)
box.add(msg)
else {
var id = msg.from_id
if(!id){ // Weird zombie message!
data.log('Zombie Message:',msg.toPrintable())
return box
} else { // Regular message
var date = moment.unix(msg.date).format('DD-MM-YYYY H:mm')
name = data.getName(id,'user')
var txt = (name || id)+' {|} {grey-fg}'+date+'{/grey-fg}\n'
if(msg.media){
if(msg.media.photo)
txt += '{grey-fg}>>>{/grey-fg} (Photo)'
else if(msg.media.audio)
txt += "{grey-fg}>>>{/grey-fg} (Audio Message) "+msg.media.audio.duration+" seconds"
else if(!msg.message)
txt += "{grey-fg}>>>{/grey-fg} (Unsupported Message)"
}
if(msg.message) txt += '{grey-fg}>{/grey-fg} '+msg.message
if(prepend) box.prepend(txt)
else box.add(txt)
2015-08-31 03:03:28 +02:00
}
}
// Mark messages as read if needed
if(param === data.selectedWindow) data.markAsRead(param)
return box
2015-08-30 01:49:03 +02:00
}
// - Entry Point -
// Load authKey and userdata from disk, then act depending on outcome
data.load('Starting up...')
2015-08-31 14:04:32 +02:00
data.screen.render()
2015-08-31 18:37:32 +02:00
fs.exists(data.keyFile,function(exists){
2015-08-29 02:38:59 +02:00
if(exists){
//log('Authorization Key found')
2015-08-31 18:37:32 +02:00
fs.readFile(data.keyFile,function(err,content){
2015-08-29 02:38:59 +02:00
if(err)
2015-08-31 14:04:32 +02:00
data.log('Error while reading key:',err)
2015-08-29 02:38:59 +02:00
else {
2015-08-31 14:04:32 +02:00
data.app.authKey = data.telegramLink.retrieveAuthKey(content,'password') // yeah sorry just testing
data.log('Authorization Key found')
2015-08-31 18:37:32 +02:00
fs.readFile(data.userFile,function(err,res){
if(err)
2015-08-31 14:04:32 +02:00
data.log("FATAL: couldn't read user_data.json")
else {
try {
2015-08-31 14:04:32 +02:00
data.user = JSON.parse(res)
data.log('Welcome',data.getName(data.user.id,'user'))
} catch (e) {
2015-08-31 14:04:32 +02:00
data.log("FATAL: user data corrupted:",e)
}
2015-08-31 14:04:32 +02:00
data.connect()
}
})
2015-08-29 02:38:59 +02:00
}
})
} else {
2015-08-31 14:04:32 +02:00
data.connect()
2015-08-29 02:38:59 +02:00
}
})