golang客户端上传文件

文章介绍了如何使用Docker部署MinIO对象存储服务,并通过Golang编写一个HTTP服务来处理文件上传请求,包括错误处理和静态文件服务。同时,提供了前端HTML和JavaScript代码示例,展示如何使用AJAX进行文件上传。
摘要由CSDN通过智能技术生成

参考官网安装minio

# https://min.io/docs/minio/container/index.html

mkdir -p ~/minio/data
docker run \
   -p 9000:9000 \
   -p 9090:9090 \
   --name minio \
   -v ~/minio/data:/data \
   -e "MINIO_ROOT_USER=ROOTNAME" \
   -e "MINIO_ROOT_PASSWORD=CHANGEME123" \
   quay.io/minio/minio server /data --console-address ":9090"
   
#或者使用我的镜像,默认用户密码看日志即可
docker run -d --name minio -p 9000:9000 -p 9001:9001 registry.cn-shenzhen.aliyuncs.com/jbjb/csi:mino server /data --console-address ":9001"
  • golang代码

  • (该代码使用 log 包记录错误日志,并在出现错误时返回适当的 HTTP 状态码和消息,以便客户端显示相应的提示信息。同时,添加了一个处理静态文件请求的处理器,并将前端页面文件放在 ./static/ 目录下。)
package main

import (
	"fmt"
	"net/http"
        "net/url"
	"log"
	"strings"
	"github.com/minio/minio-go/v7"
	"github.com/minio/minio-go/v7/pkg/credentials"
)

const (
	endpoint       = "172.17.0.2:9000"
	accessKeyID    = "minioadmin"
	secretAccessKey = "minioadmin"
	useSSL         = false
	bucketName     = "jsj"
)

func main() {
	// 初始化 MinIO 客户端对象
	minioClient, err := minio.New(endpoint, &minio.Options{
		Creds:  credentials.NewStaticV4(accessKeyID, secretAccessKey, ""),
		Secure: useSSL,
	})
	if err != nil {
		panic(err)
	}

	// 处理文件上传请求
	http.HandleFunc("/upload", func(w http.ResponseWriter, r *http.Request) {
		// 获取上传的文件
		file, header, err := r.FormFile("file")
		if err != nil {
			log.Printf("get form err: %s", err.Error()) // 记录错误日志
			http.Error(w, "Bad Request", http.StatusBadRequest)
			return
		}
		defer file.Close()

		// 将文件上传到 MinIO 中
		objectName := header.Filename
		contentType := "application/octet-stream"

		objectName = strings.ReplaceAll(objectName, "@", "_") // 替换 @ 符号
		encodedObjectName := url.QueryEscape(objectName)
		n, err := minioClient.PutObject(r.Context(), bucketName, encodedObjectName, file, -1, minio.PutObjectOptions{ContentType: contentType})
		if err != nil {
			log.Printf("upload file err: %s", err.Error()) // 记录错误日志
			http.Error(w, "Internal Server Error", http.StatusInternalServerError)
			return
		}

		w.WriteHeader(http.StatusOK)
		fmt.Fprintf(w, "File uploaded (%d bytes)", n)
	})

	// 处理静态文件请求
	staticHandler := http.FileServer(http.Dir("./static"))
	http.Handle("/", staticHandler)

	// 启动 HTTP 服务器
	fmt.Println("Starting server on port :8080")
	http.ListenAndServe(":8080", nil)
}
  • index.html文件

  • (以下是一个简单的 HTML 和 JavaScript 代码示例,展示了如何使用 AJAX 实现文件上传和错误信息提示:)
<!DOCTYPE html>
<html>
<head>
	<meta charset="utf-8">
	<title>Upload a file</title>
</head>
<body>
	<h1>Upload a file</h1>
	<form id="upload-form" enctype="multipart/form-data">
		<input type="file" name="file"><br><br>
		<button type="submit">Upload</button>
	</form>

	<div id="message"></div>

	<script>
		// 获取表单元素和提示信息元素
		var form = document.getElementById('upload-form');
		var message = document.getElementById('message');

		// 监听提交事件,使用 AJAX 发送文件上传请求
		form.addEventListener('submit', function(event) {
			event.preventDefault(); // 阻止表单默认提交行为

			var xhr = new XMLHttpRequest();
			xhr.open('POST', '/upload');
			xhr.onreadystatechange = function() {
				if (xhr.readyState === XMLHttpRequest.DONE) {
					if (xhr.status === 200) {
						message.textContent = 'File uploaded successfully.';
					} else {
						message.textContent = 'File upload failed: ' + xhr.statusText;
					}
				}
			};
			xhr.onerror = function() {
				message.textContent = 'File upload failed. Please check your internet connection and try again.';
			};
			var formData = new FormData();
			formData.append('file', form.file.files[0], encodeURIComponent(form.file.files[0].name));
			xhr.send(formData);
		});
	</script>
</body>
</html>

最后运行

 go mod init  xxxx
 go mod tidy
 go run xxx.go

在这里插入图片描述
在这里插入图片描述

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
要实现 Golang 的文件分块上传逻辑,可以按照以下步骤进行: 1. 客户端上传文件分成若干块,每块的大小可以根据实际情况进行设置。同时,客户端需要记录文件的总大小和每个块的编号,用于服务端进行合并。 2. 客户端通过 HTTP 请求将每个块上传到服务端,每个块都需要指定块的编号和文件的总大小。服务端需要根据块的编号进行排序,然后将所有块的内容按顺序合并成完整的文件。 3. 服务端可以使用 Golang 的 multipart 包来处理分块上传的请求,multipart 包可以自动解析 HTTP 请求中的块编号、总大小和内容,并将内容写入到文件中。 下面是一个简单的分块上传示例代码: ```go const ( maxChunkSize = 10 * 1024 * 1024 // 每个块的大小为 10MB ) func uploadHandler(w http.ResponseWriter, r *http.Request) { // 解析 HTTP 请求 err := r.ParseMultipartForm(32 << 20) // 限制上传文件大小为 32MB if err != nil { http.Error(w, err.Error(), http.StatusInternalServerError) return } // 获取文件总大小和块编号 totalSize, _ := strconv.Atoi(r.FormValue("totalSize")) chunkIndex, _ := strconv.Atoi(r.FormValue("chunkIndex")) // 获取当前块的内容 file, _, err := r.FormFile("file") if err != nil { http.Error(w, err.Error(), http.StatusInternalServerError) return } defer file.Close() // 读取块内容 chunkSize := maxChunkSize if chunkIndex == totalChunks-1 { chunkSize = totalSize - (totalChunks-1)*maxChunkSize } chunk := make([]byte, chunkSize) _, err = file.Read(chunk) if err != nil { http.Error(w, err.Error(), http.StatusInternalServerError) return } // 将块内容写入到文件中 f, err := os.OpenFile("upload/" + r.FormValue("fileName"), os.O_WRONLY|os.O_CREATE, 0666) if err != nil { http.Error(w, err.Error(), http.StatusInternalServerError) return } defer f.Close() _, err = f.WriteAt(chunk, int64(chunkIndex*maxChunkSize)) if err != nil { http.Error(w, err.Error(), http.StatusInternalServerError) return } // 返回上传成功的响应 w.WriteHeader(http.StatusOK) w.Write([]byte("Upload Success")) } ``` 在客户端,可以使用类似下面的代码来实现文件分块上传: ```go func uploadFile(filepath string) error { // 打开文件并计算块数量 f, err := os.Open(filepath) if err != nil { return err } defer f.Close() fileInfo, err := f.Stat() if err != nil { return err } totalSize := fileInfo.Size() totalChunks := int(math.Ceil(float64(totalSize) / float64(maxChunkSize))) // 分块上传文件 for i := 0; i < totalChunks; i++ { chunkSize := maxChunkSize if i == totalChunks-1 { chunkSize = int(totalSize) - i*maxChunkSize } // 创建 HTTP 请求并上传块内容 req, err := http.NewRequest("POST", "http://localhost:8080/upload", nil) if err != nil { return err } formData := url.Values{} formData.Set("fileName", filepath) formData.Set("totalSize", strconv.FormatInt(totalSize, 10)) formData.Set("chunkIndex", strconv.Itoa(i)) body := &bytes.Buffer{} writer := multipart.NewWriter(body) part, err := writer.CreateFormFile("file", filepath) if err != nil { return err } io.CopyN(part, f, int64(chunkSize)) writer.Close() req.Header.Set("Content-Type", writer.FormDataContentType()) req.Body = ioutil.NopCloser(body) client := &http.Client{} resp, err := client.Do(req) if err != nil { return err } defer resp.Body.Close() if resp.StatusCode != http.StatusOK { return fmt.Errorf("Upload Failed: %s", resp.Status) } } return nil } ``` 这样就可以实现 Golang 的文件分块上传逻辑了。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值