- vừa được xem lúc

[Golang] Lưu Ý Khi Dùng Transaction Database

0 0 25

Người đăng: Open Dev

Theo Viblo Asia

Mình hy vọng sau khi đọc bài viết này, bạn có thể tránh được những lỗi tương tự trong dự án của chính mình. Mặc dù chỉ là một lỗi rất nhỏ nhưng nó có thể khiến bạn sẽ phải hối hận.

Chúng ta sẽ đối mặt là một transaction database đơn giản. Và hầu như tất cả các develop điều biết transaction , nếu vẫn thắc mắc về nó thì kham khảo here.

Trong service sẽ làm chức năng quản lý user trong database, get user bằng id sau đó cập nhật full-name

Thứ tự các bước như sau:

  • bắt đầu 1 transaction
  • lấy thông tin user bằng id
  • Xác nhận user vẫn còn trạng thái active
  • nếu hợp lệ thì cập nhật fullname
  • commit transaction
  • nếu không hợp lệ hoặc error thì revert transaction

| Mình sẽ dùng gorm để truy cập database

  • đoạn code repository get user bằng id:
func (r *userService) GetUserById(tx *gorm.DB, id int) (model.User, error) { var ( user model.User err error ) err = tx.Table(user.Table()).Where("id = ?", id).First(&user).Error if err != nil { return user, err } return user, nil
}
  • code logic:
func (s *service) UpdateUserId(ctx context.Context, id int) (error) { tx := r.gormDB.Begin() defer func() { if err != nil { tx.Rollback() } }() user, err := s.userService.GetUserById(tx, id) if err != nil { return err } if user.Status != "active" { return nil } user.UpdatedAt = time.Now() err = s.userService.UpdateName(tx, name) if err != nil { return err } err = tx.Commit() if err != nil { return err } return nil
}

Vấn Đề

  • tất cả các request từ client sẽ bị time-out, database sẽ thêm rất nhiều connection vào hệ thống.
  • không có bất cứ 1 request nào có thể hoàn thành và kết thúc.

Tại Sao vấn đề xảy ra đó là khi user có status khác active điều này có nghĩa là sẽ return nil và transaction sẽ không release lock trên record của user đó.

mặc dù trong function defer có gọi rollback() nhưng chỉ xảy ra khi có error, user khác active sẽ return nil và record sẽ bị lock cho đến khi transaction bị timeout.

Fix Để fix vấn đề trên là cực kì đơn giản, thêm rollback().

if user.Status != "active" { tx.Rollback() return nil
}

còn nếu bạn vẫn muốn rollback() trong defer thì sẽ return error.

var ErrUserNotActive = errors.New("user not active")
func (s *service) UpdateUserId(ctx context.Context, id int) (error) { tx := r.gormDB.Begin() defer func() { if err != nil { tx.Rollback() } }() user, err := s.userService.GetUserById(tx, id) if err != nil { return err } if user.Status != "active" { return ErrUserNotActive } user.UpdatedAt = time.Now() err = s.userService.UpdateName(tx, name) if err != nil { return err } err = tx.Commit() if err != nil { return err } return nil
}

Cảm ơn các bạn đã đọc. Có gì liên quan đến golang thì liên hệ mình:

Contact

Bình luận

Bài viết tương tự

- vừa được xem lúc

gRPC - Nó là gì và có nên sử dụng hay không?

Nhân một ngày rảnh rỗi, mình ngồi đọc lại RPC cũng như gRPC viết lại để nhớ lâu hơn. Vấn đề là gì và tại sao cần nó .

0 0 132

- vừa được xem lúc

Embedded Template in Go

Getting Start. Part of developing a web application usually revolves around working with HTML as user interface.

0 0 57

- vừa được xem lúc

Tạo Resful API đơn giản với Echo framework và MySQL

1. Giới thiệu.

0 0 61

- vừa được xem lúc

Sử dụng goquery trong golang để crawler thông tin các website Việt Nam bị deface trên mirror-h.org

. Trong bài viết này, mình sẽ cùng mọi người khám phá một package thu thập dữ liệu có tên là goquery của golang. Mục tiêu chính của chương trình crawler này sẽ là lấy thông tin các website Việt Nam bị deface (là tấn công, phá hoại website, làm thay đổi giao diện hiển thị của một trang web, khi người

0 0 237

- vừa được xem lúc

Tạo ứng dụng craw dữ liệu bing với Golang, Mysql driver

Chào mọi người . Lâu lâu ta lại gặp nhau 1 lần, để tiếp tục series chia sẻ kiến thức về tech, hôm nay mình sẽ tìm hiểu và chia sẻ về 1 ngôn ngữ đang khá hot trong cộng đồng IT đó là Golang.

0 0 76

- vừa được xem lúc

Golang: Rest api and routing using MUX

Routing with MUX. Let's create a simple CRUD api for a blog site. # All . GET articles/ .

0 0 55