## This file defines programs, and HOW those
## programs are run.
##
## Look at the definition of "totem" for a
## fully documented example.
module Application
## def totem(files) starts the definition of totem
def totem(files)
## this is the `case' statement.
## it looks up the mode and does different
## things depending on the mode.
case files.mode
## if the mode is 0 (default)
when 0
## start totem in fullscreen
"totem --fullscreen #{files}"
## if the mode is 1
when 1
## start totem normally
"totem #{files}"
end
## the mode is a number which you type in
## between two r's. for example, press:
## r1r
## to run the file in mode 1.
## there are shortcuts: the key l runs in mode 0,
## and L in mode 1.
## the variable `files' is of a type RunContext,
## see ranger/code/runcontext.rb for details.
end
def aunpack(files)
case files.mode
when 0; "aunpack #{files}"
when 1; "aunpack -l #{files} | less"
end
end
def gedit(files)
"gedit #{files}"
end
def mplayer(files)
case files.mode
when 0; "mplayer -fs -sid 0 #{files}"
when 1; "mplayer -sid 0 #{files}"
when 2; "mplayer -vm sdl -sid 0 #{files}"
when 3; "mplayer -mixer software #{files}"
else nil end
end
def mplayer_detached(files)
files.base_flags = 'd'
mplayer(files)
end
def evince(files)
"evince #{files}"
end
def feh(files)
case files.mode
when 0; "feh #{files}"
when 1; "feh --bg-scale #{files.one}"
when 2; "feh --bg-tile #{files.one}"
when 3; "feh --bg-center #{files.one}"
when 4; "gimp #{files}"
when 5; "feh -F #{files}"
else nil end
end
def interpreted_language(files)
case files.mode
when 1; run(files)
when 0; vi(files)
else nil end
end
def zsnes(files)
"zsnes #{files.first}"
end
def run(files)
files.first.executable? ? "#{files.one}" : nil
end
def vi_or_run(files)
case files.mode
when 1; run(files)
when 0; vi(files)
else nil end
end
def vi(files)
commands = [
'map h :quit<cr>',
'map q h',
'map H :unmap h<CR>:unmap H<CR>:unmap q<CR>',
].map package main
import (
"bytes"
"encoding/json"
"flag"
"fmt"
"io/ioutil"
"log"
"os"
"os/exec"
"strings"
"time"
"github.com/lrstanley/girc"
)
// function to grease error checking
func checkerr(err error) {
if err != nil {
panic(err)
}
}
//Conf ... right now Conf.Pass isn't used,
//but i'm leaving it so the bot
//can have a registered nick
type Conf struct {
Owner string
Chan string
Server string
Port int
Nick string
Pass string
User string
Name string
SSL bool
}
func main() {
//check for config file specified by command line flag -c
jsonlocation := flag.String("c", "config.json", "Path to config file in JSON format")
//spit out config file structure if requested
jsonformat := flag.Bool("j", false, "Describes JSON config file fields")
flag.Parse()
if *jsonformat == true {
fmt.Println(`Here is the format for the JSON config file:
{
"owner": "YourNickHere",
"chan": "#favchannel",
"server": "irc.freenode.net",
"port": 6697,
"nick": "goofbot",
"pass": "",
"user": "goofbot",
"name": "Goofus McBotus",
"ssl": true
}`)
os.Exit(0)
}
//read the config file into a byte array
jsonconf, err := ioutil.ReadFile(*jsonlocation)
checkerr(err)
// unmarshal the json byte array into struct conf
var conf Conf
err = json.Unmarshal(jsonconf, &conf)
checkerr(err)
// CLIENT CONFIG
client := girc.New(girc.Config{
Server: conf.Server,
Port: conf.Port,
Nick: conf.Nick,
User: conf.User,
Name: conf.Name,
Out: os.Stdout,
SSL: conf.SSL,
})
client.Handlers.Add(girc.CONNECTED, func(c *girc.Client, e girc.Event) {
//authenticate with nickserv if pass is set in config file
if conf.Pass != "" {
c.Cmd.Message("nickserv", "identify "+conf.Pass)
time.Sleep(500 * time.Millisecond)
}
//join initial channel specified in config.json
c.Cmd.Join(conf.Chan)
})
// basic command-response handler
client.Handlers.Add(girc.PRIVMSG, func(c *girc.Client, e girc.Event) {
if strings.HasPrefix(e.Last(), "!hello") {
c.Cmd.ReplyTo(e, "henlo good fren!!")
return
}
// check if the command was issued by a specific person before dying
// i had to delve into girc/event.go to find e.Source.Name
if strings.HasPrefix(e.Last(), "die, devil bird!") && e.Source.Name == conf.Owner {
c.Cmd.Reply(e, "SQUAWWWWWK!!")
time.Sleep(100 * time.Millisecond)
c.Close()
return
}
//another basic command/response. required information for the tildeverse
if strings.HasPrefix(e.Last(), "!botlist") {
c.Cmd.Reply(e, "Creator: ~a h r i m a n~ :: I'm the assistance bot for tilde.institute. Commands: !hello !join !uptime !users")
return
}
// when requested by owner, join channel specified
if strings.HasPrefix(e.Last(), "!join") && e.Source.Name == conf.Owner {
dest := strings.Split(e.Params[1], " ")
c.Cmd.Reply(e, "Right away, cap'n!")
time.Sleep(100 * time.Millisecond)
c.Cmd.Join(dest[1])
}
// respond with uptime / load
if strings.HasPrefix(e.Last(), "!uptime") {
uptime := exec.Command("uptime")
var out bytes.Buffer
uptime.Stdout = &out
err := uptime.Run()
if err != nil {
log.Fatalln("Error while running 'uptime'")
}
c.Cmd.Reply(e, out.String())
return
}
// respond with currently connected users
// TODO: prepend names with _ to avoid pings in irc
if strings.HasPrefix(e.Last(), "!users") {
users := exec.Command("who", "-q")
var out bytes.Buffer
users.Stdout = &out
err := users.Run()
if err != nil {
log.Fatalln("Error while running 'who -q'")
}
c.Cmd.Reply(e, out.String())
return
}
// number of total users
// bot dies when i run this
// TODO: defuckulate this command
if strings.HasPrefix(e.Last(), "!totalusers") {
users := exec.Command("ls /home | wc -w")
var out bytes.Buffer
users.Stdout = &out
err := users.Run()
if err != nil {
log.Fatalln("Error while running 'ls /home | wc -w'")
}
c.Cmd.Reply(e, out.String())
}
})
// die if there's a connection error
if err := client.Connect(); err != nil {
log.Fatalf("an error occurred while attempting to connect to %s: %s", client.Server(), err)
}
//TODO: figure out sigint handling
// ctrlc := make(chan os.Signal, 1)
// signal.Notify(ctrlc, os.Interrupt)
// go func() {
// <-ctrlc
// client.Close()
// os.Exit(1)
// }()
}