Истории успеха наших клиентов — лучшие проекты
Вход/ Регистрация

Как создать и развернуть приложение на Gin: быстрый и простой деплой с App Platform

1011
14 минут чтения
Средний рейтинг статьи: 5

Gin — это высокоэффективный HTTP-веб-фреймворк, написанный на языке программирования Go, который предоставляет разработчикам мощные инструменты для создания веб-приложений, RESTful API и микросервисов. Он выделяется среди других фреймворков своей высокой скоростью обработки запросов, гибкостью настройки и простотой использования.

Одним из ключевых преимуществ Gin является его производительность. Gin использует минималистичный подход к обработке HTTP-запросов, что позволяет ему быть одним из самых быстрых фреймворков на рынке. Он основан на модуле net/http стандартной библиотеки Go, что обеспечивает отличную интеграцию с существующей экосистемой Go и позволяет использовать возможности конкурентного программирования Go для обработки большого количества одновременных запросов.

Другим важным преимуществом Gin является его простота. Синтаксис и структура Gin интуитивно понятны, что снижает порог вхождения для разработчиков и ускоряет процесс разработки. Встроенная система маршрутизации позволяет легко определять и обрабатывать маршруты, а мощная система middlewares позволяет гибко управлять обработкой запросов.

Гибкость Gin также заслуживает особого внимания. Он предоставляет возможность расширения функциональности через плагины и middlewares, что позволяет адаптировать его под специфические требования проекта. Встроенная поддержка JSON и других форматов данных упрощает создание RESTful API, а инструменты для работы с запросами и ответами позволяют легко управлять данными.

В дополнение к этому, Gin обладает активным сообществом и хорошей документацией, что делает его отличным выбором для разработчиков, ищущих надежный и поддерживаемый фреймворк. Существует множество ресурсов, включая примеры кода, руководства и библиотеки, которые облегчают процесс обучения и разработки.

Создание приложения

Описание функционала

Наше приложение будет поддерживать основные CRUD-операции (создание, чтение, обновление, удаление) для заметок через RESTful API. В ходе разработки мы обсудим ключевые аспекты интеграции Gin и ORM-библиотеки GORM, а также продемонстрируем, как обеспечить безопасность и производительность нашего веб-приложения. Основные возможности нашего приложения включают:

  1. Создание новой заметки:

    • Пользователь сможет добавить новую заметку, отправив POST-запрос с заголовком и содержимым заметки.
    • Приложение сохранит новую заметку в базе данных и вернет её уникальный идентификатор.
  1. Получение всех заметок:

    • Пользователь сможет запросить список всех заметок, отправив GET-запрос.
    • Приложение вернет все заметки из базы данных в формате JSON.
  1. Получение заметки по ID:
    • Пользователь сможет получить конкретную заметку по её идентификатору, отправив GET-запрос с указанным ID.
    • Приложение найдет заметку в базе данных и вернет её в формате JSON.
  1. Обновление существующей заметки:

    • Пользователь сможет обновить существующую заметку, отправив PUT-запрос с новым заголовком и содержимым заметки.
    • Приложение обновит данные заметки в базе данных и вернет обновленную заметку.
  1. Удаление заметки:

    • Пользователь сможет удалить заметку по её идентификатору, отправив DELETE-запрос с указанным ID.
    • Приложение удалит заметку из базы данных и вернет статус успешного выполнения операции.

App Platform

Готовое окружение для быстрого деплоя
и тестирования проектов из GitHub, GitLab, Bitbucket
или любого другого git-репозитория.

Подготовка проекта

Подразумевается, что у вас установлен Go версии 1.22 (установить его можно, например, по одной из этих инструкций Windows, Ubuntu, MacOS). При использовании более ранних версий, в процессе запуска проекта могут возникнуть ошибки. Также у вас есть базовое представление о Git и аккаунт на одном из сервисов хостинга Git репозиториев (GitHub, GitLab, Bitbucket, Gitea и т.п.). 

  1. Создадим директорию проекта командой:

    
mkdir GinTW

И перейдем в нее, выполнив:

    
cd GinTW
  1. Инициируем новый модуль Go командой: 

    
go mod init gin-notes-api
  1. Установим необходимые для проекта пакеты: Gin, GORM и SQLite для работы с базой данных, командами:

    
go get -u github.com/gin-gonic/gin go get -u gorm.io/gorm go get -u gorm.io/driver/sqlite
  1. Создадим структуру проекта. Она должна выглядеть вот так:

    
GinTW/ ├── go.mod ├── main.go ├── models/ │ └── note.go ├── handlers/ │ └── note_handlers.go ├── storage/ │ └── storage.go │ └── database.go

Создать структуру вы можете средствами IDE или выполнив следующую команду в терминале:

    
mkdir -p models handlers storage && touch go.mod main.go models/note.go handlers/note_handlers.go storage/storage.go storage/database.go

Структура приложения

  • models/note.go

Определяет структуру данных для заметок. Модель Note описывает поля заметки и используется для работы с базой данных через ORM-библиотеку GORM.

    
package models // Определение структуры заметки type Note struct { ID int `json:"id" gorm:"primaryKey;autoIncrement"` // Уникальный идентификатор заметки, автоматически инкрементируемый Title string `json:"title"` // Заголовок заметки Content string `json:"content"` // Содержание заметки }
  • storage/database.go

Этот файл содержит функции для инициализации базы данных и получения экземпляра базы данных. Здесь используется ORM-библиотека GORM для работы с базой данных SQLite.

    
package storage import ( // Импортирование необходимых пакетов "gorm.io/driver/sqlite" // Драйвер для работы с SQLite "gorm.io/gorm" // Основная библиотека GORM для работы с ORM "gin-notes-api/models" // Импортирование пакета с определением моделей данных ) // Объявление глобальной переменной для хранения экземпляра базы данных var db *gorm.DB // Функция инициализации базы данных func InitDatabase() error { var err error // Открытие подключения к базе данных SQLite с использованием GORM db, err = gorm.Open(sqlite.Open("notes.db"), &gorm.Config{}) if err != nil { // Возвращение ошибки, если подключение не удалось return err } // Автоматическое создание таблицы для модели Note, если она еще не существует return db.AutoMigrate(&models.Note{}) } // Функция для получения экземпляра базы данных func GetDB() *gorm.DB { // Возвращение глобальной переменной db, содержащей подключение к базе данных return db }
  • storage/storage.go

Этот код обеспечивает выполнение основных операций CRUD (создание, чтение, обновление, удаление) для модели Note, используя GORM для взаимодействия с базой данных SQLite.

    
package storage import ( "gin-notes-api/models" // Импортирование пакета с определением моделей данных ) // Функция для получения всех заметок из базы данных func GetAllNotes() []models.Note { var notes []models.Note // Использование GORM для выполнения SQL-запроса SELECT и заполнения среза notes db.Find(¬es) return notes // Возвращение всех найденных заметок } // Функция для получения заметки по ID func GetNoteByID(id int) *models.Note { var note models.Note // Использование GORM для выполнения SQL-запроса SELECT с условием WHERE id = id заметки if result := db.First(¬e, id); result.Error != nil { return nil // Возвращение nil, если заметка с указанным ID не найдена } return ¬e // Возвращение найденной заметки } // Функция для создания новой заметки func CreateNote(title, content string) models.Note { note := models.Note{ Title: title, Content: content, } // Использование GORM для выполнения SQL-запроса INSERT и сохранения новой заметки в базе данных db.Create(¬e) return note // Возвращение созданной заметки } // Функция для обновления существующей заметки по ID func UpdateNote(id int, title, content string) *models.Note { var note models.Note // Использование GORM для выполнения SQL-запроса SELECT с условием WHERE id = id заметки if result := db.First(¬e, id); result.Error != nil { return nil // Возвращение nil, если заметка с указанным ID не найдена } note.Title = title note.Content = content // Использование GORM для выполнения SQL-запроса UPDATE и сохранения обновленной заметки в базе данных db.Save(¬e) return ¬e // Возвращение обновленной заметки } // Функция для удаления заметки по ID func DeleteNoteByID(id int) bool { // Использование GORM для выполнения SQL-запроса DELETE с условием WHERE id = id заметки if result := db.Delete(&models.Note{}, id); result.Error != nil { return false // Возвращение false, если удаление не удалось } return true // Возвращение true при успешном удалении заметки }
  • handlers/note_handlers.go

Этот файл содержит функции-обработчики (хендлеры) для обработки HTTP-запросов. Эти функции вызываются в ответ на различные маршруты и выполняют действия, такие как создание, получение, обновление и удаление заметок.

    
package handlers import ( "net/http" // Пакет для работы с HTTP "strconv" // Пакет для конвертации строк в другие типы данных "github.com/gin-gonic/gin" // Веб-фреймворк Gin "gin-notes-api/storage" // Импортирование модуля для работы с базой данных ) // Обработчик для получения всех заметок func GetNotes(c *gin.Context) { notes := storage.GetAllNotes() // Получение всех заметок из хранилища c.JSON(http.StatusOK, notes) // Возвращение заметок в формате JSON с кодом 200 (OK) } // Обработчик для получения заметки по ID func GetNoteByID(c *gin.Context) { // Конвертация параметра ID из строки в целое число id, err := strconv.Atoi(c.Param("id")) if err != nil { // Возвращение ошибки 400 (Bad Request), если ID некорректен c.JSON(http.StatusBadRequest, gin.H{"error": "Invalid note ID"}) return } // Получение заметки по ID из хранилища note := storage.GetNoteByID(id) if note == nil { // Возвращение ошибки 404 (Not Found), если заметка не найдена c.JSON(http.StatusNotFound, gin.H{"error": "Note not found"}) return } // Возвращение найденной заметки в формате JSON с кодом 200 (OK) c.JSON(http.StatusOK, note) } // Обработчик для создания новой заметки func CreateNote(c *gin.Context) { // Структура для хранения входных данных var input struct { Title string `json:"title" binding:"required"` Content string `json:"content" binding:"required"` } // Привязка входных данных в формате JSON к структуре input if err := c.ShouldBindJSON(&input); err != nil { // Возвращение ошибки 400 (Bad Request), если входные данные некорректны c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()}) return } // Создание новой заметки в хранилище note := storage.CreateNote(input.Title, input.Content) // Возвращение созданной заметки в формате JSON с кодом 201 (Created) c.JSON(http.StatusCreated, note) } // Обработчик для обновления существующей заметки по ID func UpdateNoteByID(c *gin.Context) { // Конвертация параметра ID из строки в целое число id, err := strconv.Atoi(c.Param("id")) if err != nil { // Возвращение ошибки 400 (Bad Request), если ID некорректен c.JSON(http.StatusBadRequest, gin.H{"error": "Invalid note ID"}) return } // Структура для хранения входных данных var input struct { Title string `json:"title" binding:"required"` Content string `json:"content" binding:"required"` } // Привязка входных данных в формате JSON к структуре input if err := c.ShouldBindJSON(&input); err != nil { // Возвращение ошибки 400 (Bad Request), если входные данные некорректны c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()}) return } // Обновление заметки в хранилище note := storage.UpdateNote(id, input.Title, input.Content) if note == nil { // Возвращение ошибки 404 (Not Found), если заметка не найдена c.JSON(http.StatusNotFound, gin.H{"error": "Note not found"}) return } // Возвращение обновленной заметки в формате JSON с кодом 200 (OK) c.JSON(http.StatusOK, note) } // Обработчик для удаления заметки по ID func DeleteNoteByID(c *gin.Context) { // Конвертация параметра ID из строки в целое число id, err := strconv.Atoi(c.Param("id")) if err != nil { // Возвращение ошибки 400 (Bad Request), если ID некорректен c.JSON(http.StatusBadRequest, gin.H{"error": "Invalid note ID"}) return } // Удаление заметки из хранилища if success := storage.DeleteNoteByID(id); !success { // Возвращение ошибки 404 (Not Found), если заметка не найдена c.JSON(http.StatusNotFound, gin.H{"error": "Note not found"}) return } // Возвращение кода 204 (No Content) при успешном удалении c.Status(http.StatusNoContent) }
  • main.go

Этот файл является основной точкой входа в приложение. В нем происходит инициализация базы данных и настройка маршрутов для обработки HTTP-запросов с использованием веб-фреймворка Gin.

    
package main import ( "log" // Пакет для логирования "github.com/gin-gonic/gin" // Веб-фреймворк Gin "gin-notes-api/handlers" // Импортирование модуля с обработчиками запросов "gin-notes-api/storage" // Импортирование модуля для работы с базой данных ) func main() { // Инициализация базы данных if err := storage.InitDatabase(); err != nil { log.Fatalf("Failed to initialize database: %v", err) // Логирование ошибки и завершение программы, если инициализация базы данных не удалась } // Создание нового роутера Gin с настройками по умолчанию router := gin.Default() // Определение маршрутов и привязка их к соответствующим обработчикам router.GET("/notes", handlers.GetNotes) // Маршрут для получения всех заметок router.GET("/notes/:id", handlers.GetNoteByID) // Маршрут для получения заметки по ID router.POST("/notes", handlers.CreateNote) // Маршрут для создания новой заметки router.PUT("/notes/:id", handlers.UpdateNoteByID) // Маршрут для обновления заметки по ID router.DELETE("/notes/:id", handlers.DeleteNoteByID) // Маршрут для удаления заметки по ID // Запуск веб-сервера на порту 8080 router.Run(":8080") }

Теперь мы можем запустить приложение локально и протестировать его функциональность.

Для запуска используем команду:

    
go run main.go

Примеры curl-запросов для тестирования функционала

Создать новую заметку:

    
curl -X POST http://localhost:8080/notes -H "Content-Type: application/json" -d '{"title":"Заголовок","content":"Тело заметки"}'

Получить все заметки:

    
curl -X GET http://localhost:8080/notes

Получить заметку по ID:

    
curl -X GET http://localhost:8080/notes/1

Обновить заметку по ID:

    
curl -X PUT http://localhost:8080/notes/1 -H "Content-Type: application/json" -d '{"title":"Обновленный заголовок","content":"Обновленное тело заметки"}'

Удалить заметку по ID:

    
curl -X DELETE http://localhost:8080/notes/1

Image1

Деплой приложения в Timeweb Cloud App Platform

Для развертывания приложения с помощью App Platform, необходимо разместить проект в git-репозитории. В данном примере используется GitHub.

Создание и загрузка репозитория

Создайте новый репозиторий на GitHub и инициализируйте его локально в директории вашего проекта:

    
git init -b main
git add .
git commit -m 'First commit'

Теперь можем загрузить репозиторий на удаленный сервер, воспользовавшись командами, которые указаны при создании нового репозитория github:

    
git remote add origin git@github.com:ваш_пользователь/ваш_репозиторий.git
git push -u origin main

Настройка Timeweb Cloud App Platform

  • Перейдите в раздел App Platform и нажмите «Создать». 

  • В разделе «Тип» выберите вкладку «Backend» и найдите фреймворк Gin.

  • Подключите ваш GitHub-аккаунт, предоставив доступ к репозиториям, или выберите необходимый репозиторий вручную.

Image2

  • После подключения Github-аккаунта, разделе «Репозиторий» выберите ваш репозиторий с приложением.

Image3

  • Выберите регион, в котором будет размещено приложение.

  • В разделе «Конфигурация» выберите минимальные настройки, их будет достаточно для текущего проекта. При необходимости, конфигурацию можно будет изменить позже.

  • В разделе «Настройки приложения» оставьте значения по умолчанию. В более сложных проектах можно указать переменные окружения и команду для сборки приложения.

  • Укажите имя будущего приложения и нажмите «Запустить деплой».

Процесс деплоя может занять до 10 минут. После завершения вы увидите сообщение «Deployment successfully completed» в логах деплоя.

Перейдите в раздел «Настройки» на странице приложения, где будет указан домен, привязанный к вашему приложению. 

Image4

В этом же разделе можно изменить конфигурацию сервера, отредактировать настройки деплоя и изменить привязку домена. При подключении своего домена, для него автоматически будет установлен сертификат Let’s Encrypt, который будет автоматически продлеваться за 7 дней до истечения срока.

Для проверки, что приложение работает корректно, можем выполнить curl-запрос, изменив localhost на привязанный домен:

    
curl -X GET https://ваш_домен/notes

Заключение

Мы разработали базовое веб-приложение для управления заметками с использованием фреймворка Gin и библиотеки GORM. Созданный RESTful API поддерживает основные CRUD-операции, что делает приложение простым и удобным в использовании.

Gin показал себя как эффективный и простой в освоении инструмент. Его система маршрутизации и поддержка одновременных запросов упростили процесс разработки. GORM помог легко работать с базой данных, автоматизировав многие задачи.

Приложение успешно развернуто на платформе Timeweb Cloud App Platform, что сделало процесс деплоя быстрым и надежным. App Platform предоставляет все необходимые инструменты для комфортного управления приложением.

В будущем можно добавить новые функции, такие как аутентификация пользователей и улучшенные возможности поиска заметок.

Этот проект продемонстрировал, как современные инструменты разработки, такие как Gin и GORM, могут упростить создание веб-приложений.

1011
14 минут чтения
Средний рейтинг статьи: 5
Хотите внести свой вклад?
Участвуйте в нашей контент-программе за
вознаграждение или запросите нужную вам инструкцию
img-server
Пока нет комментариев