Une semaine pour coder par soi même : Jour 6 et 7


[23/12/2018] - ~8mins - #golang #code


Bon aujourd'hui je groupe le week-end dans un seul post.

Hier ?

J'ai … rien fait.

Voilà.

Aujourd'hui ?

J'ai … débloqué le tout petit truc qui me coinçait.

Bref je pense avoir surpassé tous les ptits trucs qui me faisaient chier et maintenant j'ai un code que je comprends de partout et qui marche comme je l'intuitais.

Je suis super content.

En fait, le truc qui m'a complètement paralysé dans ma fonction connection.Interact() était tout bête : j'arrivais à récupérer le texte et donc à l'afficher, mais étrangement je ne parvenais pas à réagir à ce texte reçu.

Du coup lorsque je recevais un PING, je le voyais, je créais le PONG qui va bien, mais il partait pas.

Et j'ai buté sur ça depuis vendredi soir.

Et encore une fois cette nuit, l'illumination !

Dans ma fonction connection.Interact(), lorsque je recevais le PING, j'étais dans le case reader en train d'éxecuter la commande parseIrc, mais du coup, je n'étais plus en mesure d'écouter le chan entrant.

Bon c'est dur à expliquer mais, le truc pour débloquer ça, a été tout simplement de foutre le parseIrc dans une routine en parallèle, comme ça, pendant que ça fait son boulot, la boucle se termine et elle est de nouveau disponible en écoute.

Voilà Samedi je n'y ai pas touché à cause de ça (bon j'ai été assez pris à côté faut dire, mais du coup je n'ai pas été très motivé).

Une fois ce verrou enlevé j'ai eu envie de me refaire chier à faire des trucs tordus.

J'ai donc commencé un système antiflood, pour ralentir le texte sortant afin de ne pas se faire jarter comme un pleutre lorsqu'on balance trop de texte d'un coup.

J'utilise donc une variable que j'incrémente à chaque fois que j'envoie du texte et qui se décrémente petit à petit chaque seconde.

Bon j'ai pas encore fait tourner le truc comme il faut mais c'est une première étape.

Avant de rajouter des commandes à proprement parlé au bot, j'ai envie de gérer le texte sortant comme une FIFO et donc gérer ça avec un fichier comme j'ai fait dans le bot en bash.

Je vais donc m'attarder à regarder un peu la doc de Go pour voir comment gérer les fichiers mais ça ne devrait pas être bien compliqué.

Bon et même si à la base je ne voulais que dédier une semaine à ce projet, il est suffisamment avancé pour que je le finisse maintenant.

Vais pouvoir foutre au rencard mon Frankenscript et remplacer par ce ptit programme.

Le code

{{}}

package main

import (

"bufio"

"fmt"

"io"

"net"

"os"

"strings"

"time"

)

var server string = "localhost"

var port string = "6667"

var channel string = "#lms"

var nick string = "bab"

var onchan bool

var rouge string = "\033[1;31m"

var vert string = "\033[1;32m"

var jaune string = "\033[1;33m"

var cyan string = "\033[1;36m"

var normal string = "\033[0m"

var me string

var debug bool = false

type Ircconnection struct {

Server   string

Port     string

Nick     string

Channel  string

Bidule   bool

Conn     net.Conn

Receiver chan string

Emitter  chan string

RawEmitter	chan string

counter	 int

}

func (connection *Ircconnection) Connect() {

var err error

connection.Conn, err = net.Dial("tcp", server+":"+port)

// defer connection.Conn.Close()

fmt.Println(rouge + "Connection to " + server + ":" + port + normal)

if err != nil {

	fmt.Println(err)

	os.Exit(1)

}

connection.Receiver = make(chan string)

connection.Emitter = make(chan string)

connection.RawEmitter = make(chan string)

fmt.Println(rouge + ">> NICK " + nick + normal)

io.WriteString(connection.Conn, "NICK "+nick+"\n")

fmt.Println(rouge + ">> USER " + nick + " 0.0.0.0 " + nick + " :" + nick + " bot" + normal)

io.WriteString(connection.Conn, "USER "+nick+" 0.0.0.0 "+nick+" :"+nick+" bot\n")

go io.Copy(connection.Conn, os.Stdin)

go connection.handleIncoming()

// go connection.handleCounter()

go connection.Join(channel)

go connection.Interact()

}

func (connection *Ircconnection) Disconnect() {

connection.Conn.Close()

}

func (connection Ircconnection) Join(channel string) {

time.Sleep(1000 * time.Millisecond)

fmt.Println(rouge + ">> JOIN " + channel + normal)

io.WriteString(connection.Conn, "JOIN "+channel+"\n")

}

func (connection Ircconnection) SendMsg(msg string) {

fmt.Println(rouge + ">> PRIVMSG " + connection.Channel + " :" + msg + normal)

//io.WriteString(connection.Conn,"PRIVMSG "+connection.Channel+" :"+msg+"\n")

connection.Emitter <- msg

}

func NewIrcconnection() Ircconnection {

return Ircconnection{Server: server, Port: port, Nick: nick, Channel: channel}

}

func (connection *Ircconnection) SetNick(newnick string) {

connection.Nick = newnick

fmt.Println(rouge + "Changement de pseudo pour : " + vert + newnick + normal)

io.WriteString(connection.Conn, "NICK :"+newnick+"\n")

}

func (connection *Ircconnection) handleIncoming() {

scanner := bufio.NewScanner(connection.Conn)

go func() {

	for scanner.Scan() {

		ln := scanner.Text()

		fmt.Println(jaune + "<< " + ln + normal)

		connection.Receiver <- ln

	}

}()

}

func (connection *Ircconnection) Interact() {

for {

	select {

		case writer := <-connection.Emitter:

				connection.counter+= 200

				if debug {

				fmt.Println(cyan + writer +" "+jaune+"[",connection.counter,"]"+ normal)

				}

			go io.WriteString(connection.Conn,"PRIVMSG "+channel+" :"+writer+"\n")

		case writer := <-connection.RawEmitter:

			fmt.Println(cyan + writer + normal)

			go io.WriteString(connection.Conn,writer+"\n")

		case reader := <-connection.Receiver:

			//fmt.Println(cyan+reader+normal)

			go parseIrc(connection, reader)

	}

}

}

// Futur système anti-flood. tout pourri pour le moment.

func (connection *Ircconnection) handleCounter(){

for {

		fmt.Println("On boucle : ",connection.counter)

	time.Sleep(time.Duration(connection.counter+500) * time.Millisecond)

	if (connection.counter > 0) {

		connection.counter -= 10

	} else {

		connection.counter = 0

	}

}

}

/////////////////// MAIN //////////////////////////////////////

func main() {

if len(os.Args) > 1 {

	server = os.Args[1]

}

connection := NewIrcconnection()

connection.Connect()

in, err := net.Listen("tcp", ":4321")

defer in.Close()

if err != nil {

	fmt.Println(err)

	os.Exit(1)

}

for {

	inconn, err := in.Accept()

	if err != nil {

		fmt.Println(err)

		continue

	}

	go handleIncoming(inconn, connection)

}

}

// ------------------

// Côté IRC

// ------------------

func parseIrc(connection *Ircconnection, msg string) {

var elements []string = strings.Fields(msg)

if debug {

	for i, element := range elements {

		fmt.Print(cyan+"[", i, "|"+normal+element+cyan+"] "+normal)

	}

}

if len(elements) < 2 {

		fmt.Println("Syntax IRC ERROR !!!!")

	return

}

if elements[0] == "PING" {

	connection.RawEmitter <- "PONG :"+strings.TrimPrefix(msg, "PING :")

}

switch elements[1] {

case "421":

	fmt.Println("!! Commande non reconnue par le serveur !!")

case "433":

	fmt.Println("le pseudo déconne")

	nick = nick + "_"

	io.WriteString(connection.Conn, "NICK "+nick+"\n")

case "JOIN":

	if ":"+nick == stringCut(elements[0], "!") {

		me = strings.TrimPrefix(elements[0], ":")

		fmt.Println(me + " a rejoin le salon " + strings.Trim(elements[2], ":"))

		onchan = true

		connection.Emitter<- "Salut "+channel+" !"

	}

case "PART":

	fmt.Println("On est parti de " + strings.Trim(elements[2], ":"))

	onchan = false

case "KICK":

	if elements[3] == nick {

		fmt.Println("Ptain on s'est fait kicker de " + elements[2] + " par " + elements[4] + " !")

		io.WriteString(connection.Conn, "JOIN "+channel+"\n")

	}

case ":Closing":

		fmt.Println("\n\n\nHa et merde on est déconnecté !\n\n\n")

	// Déconnecté

	time.Sleep(10000 * time.Millisecond)

	connection.Connect()

}

if len(elements) > 3 {

	switch elements[3] {

			case ":cycle" :

			connection.Disconnect()

			connection.Connect()

		case ":heure" :

				connection.Emitter<-"Paies-toi une montre vaut rien!"+msg

	}

}

}

// ------------------

// Serveur en écoute

// ------------------

func incoming(connection Ircconnection) {

in, err := net.Listen("tcp", ":4321")

defer in.Close()

if err != nil {

	fmt.Println(err)

	os.Exit(1)

}

for {

	inconn, err := in.Accept()

	if err != nil {

		fmt.Println(err)

		continue

	}

	go handleIncoming(inconn, connection)

}

}

func handleIncoming(in net.Conn, connection Ircconnection) {

fmt.Println(vert+"Incoming from ", in.RemoteAddr(), normal)

inbuf := bufio.NewReader(in)

for {

	inmsg, err := inbuf.ReadString('\n')

	if err != nil || !onchan || inmsg == "\n" {

		break

	}

	fmt.Print(vert + "<<]] " + inmsg + normal)

	connection.SendMsg(inmsg)

	time.Sleep(500 * time.Millisecond)

}

}

// ------------

// Génériques

// ------------

func stringCut(incoming string, pattern string) string {

var results = strings.Split(incoming, pattern)

if len(results) < 1 {

	return incoming

} else {

	return results[0]

}

}

{{}}

Bon bha voilà que s'achève cette semaine de blogposts chiants.

Je vous referai ptet un article une fois terminé mais c'est tout.

Liens


=> 🏠 Retour à la home


[23/12/2018] - #golang #code


=> [>> Suivant >>] ⏭ Meta : récap 2018 | [<< Précédent <<] ⏮ Une semaine pour coder par soi même : Jour 5

Proxy Information
Original URL
gemini://lord.re/posts/150-une-semaine-de-code-jour6-7/index.gmi
Status Code
Success (20)
Meta
text/gemini
Capsule Response Time
269.088947 milliseconds
Gemini-to-HTML Time
3.631899 milliseconds

This content has been proxied by September (3851b).