Heim > Backend-Entwicklung > Golang > Golang implementiert die FTP-Funktion

Golang implementiert die FTP-Funktion

王林
Freigeben: 2023-05-15 09:58:37
Original
2338 Leute haben es durchsucht

FTP (File Transfer Protocol) ist ein gängiges Dateiübertragungsprotokoll, das zum Hoch- und Herunterladen von Dateien zwischen Servern und Clients verwendet werden kann. Die Go-Sprache ist eine Open-Source-Programmiersprache mit hoher Effizienz und Parallelitätsleistung. In diesem Artikel wird erläutert, wie Sie die Go-Sprache zum Implementieren der FTP-Funktion verwenden.

  1. Übersicht über das FTP-Protokoll

Das FTP-Protokoll basiert auf dem Client-Server-Modell, bei dem der Client eine Anfrage an den Server sendet, um Dateien hoch- oder herunterzuladen. Das FTP-Client-Programm verwendet das TCP-Protokoll, um mit dem FTP-Server zu kommunizieren, und der FTP-Server lauscht auf Port 21. Der FTP-Client verwendet unterschiedliche Portnummern für die Datenkommunikation mit dem FTP-Server. Das FTP-Protokoll wird häufig in Szenarien wie Dateifreigabe, Website-Dateiverwaltung und Sicherung verwendet.

  1. Implementieren eines FTP-Servers

Der FTP-Server muss Port 21 abhören und die vom Client gesendeten FTP-Befehle analysieren. Normalerweise sendet ein FTP-Client einen Benutzernamen und ein Passwort zur Authentifizierung an den FTP-Server. Nach erfolgreicher Authentifizierung kann der FTP-Client verschiedene FTP-Befehle ausführen, z. B. Dateien herunterladen, Dateien hochladen, Dateien löschen usw.

Das Folgende ist ein Beispielcode für die Implementierung eines FTP-Servers:

package main

import (
    "fmt"
    "net"
    "os"
    "bufio"
    "strings"
)

func main() {
    arguments := os.Args
    if len(arguments) == 1 {
        fmt.Println("Please provide a port number")
        return
    }

    PORT := ":" + arguments[1]
    l, err := net.Listen("tcp4", PORT)
    if err != nil {
        fmt.Println(err)
        return
    }
    defer l.Close()

    fmt.Println("Listening on " + PORT)

    for {
        conn, err := l.Accept()
        if err != nil {
            fmt.Println(err)
            continue
        }
        go handleConnection(conn)
    }
}

func handleConnection(conn net.Conn) {
    fmt.Println("Received connection from " + conn.RemoteAddr().String())
    conn.Write([]byte("220 Welcome to FTP Server
"))
    username := ""
    for {
        message, err := bufio.NewReader(conn).ReadString('
')
        if err != nil && err.Error() == "EOF" {
            break
        }
        fmt.Print("Message Received:", string(message))
        command := strings.TrimSpace(string(message))
        parts := strings.Split(command, " ")
        if parts[0] == "USER" {
            username = parts[1]
            conn.Write([]byte("331 Password required for " + username + "
"))
        } else if parts[0] == "PASS" {
            password := parts[1]
            if username == "admin" && password == "password" {
                conn.Write([]byte("230 Logged on
"))
            } else {
                conn.Write([]byte("530 Authentication failed
"))
            }
        } else if parts[0] == "QUIT" {
            conn.Write([]byte("221 Goodbye
"))
            conn.Close()
            break
        } else {
            conn.Write([]byte("500 Command not implemented
"))
        }
    }
}
Nach dem Login kopieren

In diesem Beispiel wird ein TCP-Server mithilfe des Net-Pakets der Standardbibliothek implementiert. Der Server lauscht auf die Portnummer und ruft die handleConnection-Funktion zur Verarbeitung auf, wenn eine neue Verbindung empfangen wird. Die handleConnection-Funktion sendet zunächst eine „220 Welcome to FTP Server“-Nachricht an den Client. Als nächstes wartet der Server darauf, dass der Client den Benutzernamen und das Passwort sendet, und liest die vom Client gesendete Nachricht, indem er die ReadString-Funktion im bufio-Paket aufruft. Wenn die empfangene Nachricht mit „USER“ beginnt, speichern Sie den Benutzernamen und senden Sie eine „331 Passwort erforderlich“-Nachricht an den Client und warten Sie darauf, dass der Client ein Passwort eingibt. Wenn die empfangene Nachricht mit „PASS“ beginnt, überprüfen Sie, ob das Passwort korrekt ist. Wenn die Überprüfung erfolgreich ist, wird die Meldung „230 Angemeldet“ an den Client gesendet, die angibt, dass sich der Benutzer erfolgreich angemeldet hat. Wenn die Passwortüberprüfung fehlschlägt, wird die Meldung „530 Authentifizierung fehlgeschlagen“ an den Client gesendet. Wenn die empfangene Nachricht mit „QUIT“ beginnt, wird eine „221 Goodbye“-Nachricht an den Client gesendet und die Verbindung zum Client geschlossen.

  1. FTP-Client implementieren

Der FTP-Client muss den Code implementieren, um mit dem FTP-Server zu kommunizieren. FTP-Clients interagieren normalerweise über die Befehlszeile oder die GUI-Schnittstelle mit Benutzern und stellen Funktionen wie Hochladen, Herunterladen, Löschen und Umbenennen bereit.

Das Folgende ist ein einfaches FTP-Client-Beispiel:

package main

import (
    "bufio"
    "fmt"
    "net"
    "os"
    "strings"
)

func main() {
    arguments := os.Args
    if len(arguments) == 1 {
        fmt.Println("Please provide a host:port string")
        return
    }

    CONNECT := arguments[1]
    c, err := net.Dial("tcp4", CONNECT)
    if err != nil {
        fmt.Println(err)
        return
    }

    for {
        message, err := bufio.NewReader(c).ReadString('
')
        if err != nil {
            fmt.Println(err)
            return
        }
        fmt.Print("Message Received:", string(message))
        if strings.Contains(strings.ToLower(message), "goodbye") {
            return
        }

        fmt.Print(">> ")
        commandReader := bufio.NewReader(os.Stdin)
        command, _ := commandReader.ReadString('
')
        fmt.Fprintf(c, command+"
")
    }
}
Nach dem Login kopieren

In diesem Beispiel wird der FTP-Client mithilfe des Net-Pakets in der Standardbibliothek implementiert. Der Client interagiert mit dem Benutzer über die Befehlszeilenschnittstelle. Zunächst stellt der Client über die Wählfunktion eine Verbindung zum FTP-Server her. Der Client empfängt dann kontinuierlich Nachrichten vom FTP-Server und gibt sie an die Befehlszeilenschnittstelle aus. Als nächstes wartet der Client darauf, dass der Benutzer einen Befehl eingibt, und sendet den Befehl über die Fprintf-Funktion an den FTP-Server. Der Client wartet dann weiterhin auf eine Antwort vom FTP-Server.

  1. Implementieren Sie das Hoch- und Herunterladen von FTP-Dateien.

Der FTP-Client kann den Befehl STOR zum Hochladen von Dateien und den Befehl RETR zum Herunterladen von Dateien verwenden. Der Befehl STOR lädt Dateien auf einen FTP-Server hoch, während der Befehl RETR Dateien von einem FTP-Server herunterlädt.

Das Folgende ist ein einfaches Beispiel für das Hoch- und Herunterladen einer FTP-Datei:

package main

import (
    "fmt"
    "io"
    "net"
    "os"
)

func main() {
    arguments := os.Args
    if len(arguments) == 1 {
        fmt.Println("Please provide a host:port string")
        return
    }

    CONNECT := arguments[1]
    c, err := net.Dial("tcp4", CONNECT)
    if err != nil {
        fmt.Println(err)
        return
    }

    for {
        var input string
        fmt.Scanln(&input)

        if input == "STOP" {
            fmt.Println("Exiting FTP Client")
            return
        }

        if input == "RETR" {
            fmt.Fprintf(c, input+"
")
            handleFileDownload(c)
        } else if input == "STOR" {
            fmt.Fprintf(c, input+"
")
            handleFileUpload(c)
        } else {
            fmt.Fprintf(c, input+"
")
            handleServerResponse(c)
        }
    }
}

func handleFileDownload(conn net.Conn) {
    var filename string
    fmt.Scanln(&filename)

    file, err := os.Create(filename)
    if err != nil {
        fmt.Println(err)
        return
    }
    defer file.Close()

    data := make([]byte, 2048)
    for {
        n, err := conn.Read(data)
        if err != nil && err != io.EOF {
            fmt.Println(err)
            return
        }
        if n == 0 {
            fmt.Println("Download complete")
            return
        }

        _, err = file.Write(data[:n])
        if err != nil {
            fmt.Println(err)
            return
        }
    }
}

func handleFileUpload(conn net.Conn) {
    var filename string
    fmt.Scanln(&filename)

    file, err := os.Open(filename)
    if err != nil {
        fmt.Println(err)
        return
    }
    defer file.Close()

    fileInfo, err := file.Stat()
    if err != nil {
        fmt.Println(err)
        return
    }

    fmt.Fprintf(conn, "%d
", fileInfo.Size())
    data := make([]byte, 2048)
    for {
        n, err := file.Read(data)
        if err != nil && err != io.EOF {
            fmt.Println(err)
            return
        }
        if n == 0 {
            fmt.Println("Upload complete")
            return
        }

        _, err = conn.Write(data[:n])
        if err != nil {
            fmt.Println(err)
            return
        }
    }
}

func handleServerResponse(conn net.Conn) {
    response := make([]byte, 2048)
    n, err := conn.Read(response)
    if err != nil {
        fmt.Println(err)
        return
    }
    fmt.Println(string(response[:n]))
}
Nach dem Login kopieren

In diesem Beispiel implementiert der FTP-Client die Befehle „STOR“ und „RETR“, um Dateien hoch- und herunterzuladen. Die Funktion handleFileUpload öffnet die Datei anhand des vom Benutzer eingegebenen Dateinamens und sendet die Dateigröße an den FTP-Server. Die Funktion liest dann den Dateiinhalt in Blöcken und sendet ihn an den FTP-Server. Die Funktion handleFileDownload empfängt Daten vom FTP-Server und schreibt sie in die neu erstellte Datei. Die Funktion handleServerResponse liest die Antwort vom FTP-Server und gibt sie auf der Konsole aus.

  1. Fazit

FTP-Server und -Client können einfach mit der Go-Sprache implementiert werden. Die Kommunikation zwischen dem FTP-Server und dem FTP-Client wird über das Net-Paket und das Bufio-Paket in der Standardbibliothek implementiert. Darüber hinaus unterstützt die Go-Sprache auch eine effiziente gleichzeitige Programmierung, wodurch die Funktionen zum Hoch- und Herunterladen von FTP-Dateien besser implementiert werden können.

Das obige ist der detaillierte Inhalt vonGolang implementiert die FTP-Funktion. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Quelle:php.cn
Erklärung dieser Website
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn
Beliebte Tutorials
Mehr>
Neueste Downloads
Mehr>
Web-Effekte
Quellcode der Website
Website-Materialien
Frontend-Vorlage