Golang задачи с собеседований
4.58K subscribers
489 photos
6 videos
12 files
256 links
Задачи, тесты и теоретические вопросы по Go. Так же по gin, grpc, горутинах, архитектуре api и http стэку.

Прислать задачу/вопрос в дар: @cyberJohnny
Сотрудничество: @cyberJohnny
加入频道
👨🏻‍💻 Golang: google wire: Как упростить поддержку кода?

Помогите пожалуйста разобраться с di google wire
wire.go

//go:build wireinject
// +build wireinject

package di

//go:generate wire ./...

import (
"github.com/3110Y/profile/internal/application/service"
"github.com/3110Y/profile/internal/application/validator"
"github.com/3110Y/profile/internal/infrastructure/database"
"github.com/3110Y/profile/internal/infrastructure/repository"
"github.com/3110Y/profile/internal/presentation/rpc"
"github.com/google/wire"
"github.com/jmoiron/sqlx"
)

type DI struct {
ProfileService *service.ProfileService
ProfileValidator *validator.ProfileValidator
ProfileRepository *repository.ProfileRepository
ProfileRPC *rpc.ProfileRPC
DB *sqlx.DB
}

func NewDI(
profileService *service.ProfileService,
profileValidator *validator.ProfileValidator,
profileRepository *repository.ProfileRepository,
profileRPC *rpc.ProfileRPC,
DB *sqlx.DB,
) *DI {
return &DI{
ProfileService: profileService,
ProfileValidator: profileValidator,
ProfileRepository: profileRepository,
ProfileRPC: profileRPC,
DB: DB,
}
}

func InitializeDI() (*DI, error) {
wire.Build(
NewDI,
wire.Bind(new(service.ProfileRepositoryInterface), new(*repository.ProfileRepository)),
wire.Bind(new(service.PasswordServiceInterface), new(*service.PasswordService)),
wire.Bind(new(rpc.ServiceProfileInterface), new(*service.ProfileService)),
service.NewPasswordService,
service.NewProfileService,
validator.NewProfileValidator,
repository.NewProfileRepository,
rpc.NewProfileRPC,
database.NewConnect,
)
return &DI{}, nil
}


Можно обойтись без struct Di?

Golang задачи с собеседований
👨🏻‍💻 Вопрос от пользователя

err_empty_response docker

Пишу веб-сервис на Golang. Хочу использовать Docker, но не получается.
app.go - главный файл

package main

import (
scraper_app "github.com/fentezi/scraper"
"github.com/fentezi/scraper/internal/handler"
"github.com/fentezi/scraper/pkg/logging"
)

func main() {
logger := logging.GetLogger()
logger.Info("Start server")
srv := new(scraper_app.Server)
logger.Info("Init handler")
hand := handler.NewHandler(logger)
if err := srv.Run("8080", hand.InitRouters()); err != nil {
logger.Fatal(err)
}
}


server.go - инициализация сервера

package scraper_app

import (
"net/http"
"time"
)

type Server struct {
httpServer *http.Server
}

func (s *Server) Run(port string, handler http.Handler) error {
s.httpServer = &http.Server{
Addr: "127.0.0.1:" + port,
Handler: handler,
ReadTimeout: 10 * time.Second,
WriteTimeout: 10 * time.Second,
}
return s.httpServer.ListenAndServe()
}

Dockerfile

FROM golang:latest

WORKDIR /app

COPY go.mod go.sum ./

RUN go mod download

COPY . .

EXPOSE 8080

CMD ["go", "run", "./cmd/app.go"]

Контейнер создался без ошибок, всё нормально. Когда в браузере ввожу "http://localhost:8080", то получаю ошибку.

Страница недоступна. Сайт localhost не отправил данных. ERR_EMPTY_RESPONSE

Как это решить?

Golang задачи с собеседований
👨🏻‍💻 Вопрос от пользователя

Создание сегмента при его объявлении

Мне всегда казалось, что сегменты не создаются при их объявлении, наверное из-за книги по которой я начинал изучать гоу «Head First. Изучаем Go » 2020 ISBN 978-5-4461-1395-8(страница 210) вот строки оттуда (картинка)
Но наткнулся на то, что всё работает и так:

package main
import "fmt"

func main() {
var strArr []string
fmt.Print(strArr)
strArr = append(strArr,"qwerty")
fmt.Print(strArr)
}

результат:

[][qwerty]

...Program finished with exit code 0
Press ENTER to exit console

Получается, что у меня устаревшая информация и книга была старого издания, либо я в чём-то заблуждаюсь. Исходя из приведённою мною примера, при объявлении сегмента, он всё-таки создаётся автоматически, так ли это?

Golang задачи с собеседований
👨🏻‍💻 Вопрос от пользователя

rune и byte в чём разница?

Зачем придумали 2 отдельных типа данных и в чём их отличие и назначение?

Golang задачи с собеседований
👨🏻‍💻 Вопрос от пользователя

Ошибка подключения приложения к mysql / GO

Разрабатываю софт для парсинга баз данных mysql, столкнулся с ошибкой при использовании github.com/go-sql-driver/mysql. При запуске софта на Windows Server 2016 вылазит такая ошибка:

[mysql] 2024/02/19 05:28:57 packets.go:36: read tcp 127.0.0.1:52891->127.0.0.1:1433: wsarecv: An existing connection was forcibly closed by the remote host.
[mysql] 2024/02/19 05:30:57 packets.go:36: read tcp 127.0.0.1:52923->127.0.0.1:1433: wsarecv: An existing connection was forcibly closed by the remote host.
Error pinging database: driver: bad connection

В чем может быть проблема?

Golang задачи с собеседований
👨🏻‍💻 Вопрос от пользователя

Организация структуры проекта с несколькими модулями Golang

В 1 проекте будут содержаться 3 различные программы: Repeater, Controller и Executor, полагаю что необходимо под каждый выделить свой модуль. И я хочу сделать 1 модуль который они будут импортировать с некоторым набором общих для них функций. Пока что максиум что вышло в каждый модуль добавить копии файла с этими общими функциями, что выглядит не очень. Я часа 4 ковырялся в статьях и видео уроках но так и не смог из 1 несчастного модуля импортировать функции в другой неменее несчастный модуль. Прошу помощи как можно подробнее обьясните как сделать то что я хочу если это вообще возможно.


Golang задачи с собеседований
👨🏻‍💻 Вопрос от пользователя

Как в go сделать переменную с двумя типами и функцию вовращающую два типа

Как сделать так чтобы в переменной left можно было хранить как NumberNode так и BinaryOperationNode, и чтобы функция могла вовращать оба эти типа?

Код:

func (p *Parser) Term() BinaryOperationNode {
left := p.Factor() // Factor возвращает NumberNode

for p.CurrentToken.Type_ == token.Number {
operation := p.CurrentToken
right := p.Factor()
left := node.BinaryOperationNode{LeftOperand: left.Value, RightOperand: right.Value, Operation: operation.Value}

}

return left
}


Golang задачи с собеседований
👨🏻‍💻 Вопрос от пользователя

После ввода команды в терминал ничего не происходит

После ввода команды (go env -w GOINSECURE=«имя проекта») в терминал ничего не происходит, в чем может быть проблема ?

Golang задачи с собеседований
👨🏻‍💻 Вопрос от пользователя

Я новичок в вебе и хочу отправить данные из моего API в другой API, но мой API отправляет, а другой почему-то не получает

func main() {
r := gin.Default()

r.POST("/send-data", func(c *gin.Context) {

var token struct {
Api_key string `json:"accsestoken"`
}
var data struct {
Phone string `json:"phone"`
Amount float64 `json:"amount"`
}
if err := c.BindJSON(&data); err != nil {
c.JSON(http.StatusBadRequest, gin.H{"error": "Invalid JSON"})
return
}

token.Api_key = tools.Create_token(data.Phone, data.Amount)
local_id := uuid.New()

payload := map[string]interface{}{
"api_key": token.Api_key,
"phone": data.Phone,
"amount": data.Amount,
"local_id": local_id.String(),
}

res, err := json.Marshal(payload)
if err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": "Failed to prepare data"})
return
}

resp, err := http.Post("http://localhost:8000/api/trxs", "application/json", bytes.NewBuffer(res))
if err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": "Failed to send data to target API"})
return
}
defer resp.Body.Close()

c.JSON(http.StatusOK, gin.H{"message": "Data sent successfully"})
})

r.Run(":3000")
}

Функция отправляющая данные:

r.POST("/api/trxs", func(c *gin.Context) {

var data struct {
Api_key string `json:"api_key"`
Phone string `json:"phone"`
Amount float64 `json:"amount"`
Local_id string `json:"local_id"`
}

if err := c.ShouldBindJSON(&data); err != nil {
c.JSON(http.StatusBadRequest, gin.H{"unsuccess": false, "error": err.Error()})
return
}
fmt.Println(data)
// Возвращение данных клиенту
c.JSON(http.StatusOK, data)
})

r.Run(":8000")

эта функция получает данные, но выдает ошибку:

{
"error": "EOF",
"unsuccess": false
}

Когда я отправляю запрос из первой функции, я получаю:

{
"message": "Data sent successfully"
}

Но вторая функция выдаёт:

{
"error": "EOF",
"unsuccess": false
}


Golang задачи с собеседований
👨🏻‍💻 Вопрос от пользователя

Есть ли брокеры сообщений где можно сохранять объекты?

Прошу не пинать если вопрос глупый.

В пет-проекте понадобился брокер сообщений. Я пробовал натс но он работает только со строками.

А есть ли такие брокеры, которые умеют работать гибридно(pub/sub+p2p) и, главное, могут сохранять не только строки а ещё и объекты(структуры, мапы и т.д). P2P в приоритете, а гибрид необязательно но это задел на будущий функционал.

Да, знаю, что можно запаковать объект в json и сохранять как строку, но это накладные расходы на распаковку/запаковку каждой транзакции.

Да, работа ведётся с транзакциями, которые один пользователь запрашивает у другого. Может кто то считает что брокеры для этого не лучший вариант и знает вариант получше?

Golang задачи с собеседований
👨🏻‍💻 Вопрос от пользователя

Как получить доступ к полям интерфейсного типа если знаю, что там за структура?

Есть

type Mover interface {
Move()
}

type M struct {
Name string
Age int
}

func doMoving(mover Mover) {
mover.Move()
// Если я знаю, что у меня под mover-ом структура M
//то как я могу получить в этом месте доступ к полям Name и Age?
// Чтобы можно было как то так:
// fmt.Println(mover.Name)
}


Golang задачи с собеседований
👨🏻‍💻 Вопрос от пользователя

Уведомлять пользователей telegram через определённый промежуток времени Golang

У меня есть телеграмм бот, который по взаимодействию с пользователем работает по типу "Запрос-ответ". Необходимой функцией этого бота является уведомлять пользователей в 9 утра о том, что нужно ввести новые данные с которыми он будет работать. Проблема заключается в том, что я не понимаю, как мне проверять текущее время периодически, поскольку код, который я использую для телеграмм бота, заканчивается входом в цикл for update := range updates , но внутри этого цикла, судя по логированию, я попадаю только, когда получаю сообщения от пользователя. Скорее всего я не до конца понимаю некоторых особенностей методов go-telegram-bot-api

func main() {
// используя токен создаем новый инстанс бота
bot, err := tgbotapi.NewBotAPI(telegramBotToken)
if err != nil {
log.Panic(err)
}
bot.Debug = true
log.Printf("Авторизация аккаунта бота %s", bot.Self.UserName)
// u - структура с конфигом для получения апдейтов
u := tgbotapi.NewUpdate(0)
u.Timeout = 1
// используя конфиг u создаем канал в который будут прилетать новые сообщения
updates := bot.GetUpdatesChan(u)
err = callAt(11, 17, 0, myfunc)
if err != nil {
fmt.Printf("error: %v\n", err)
}
//обработка сообщений
for update := range updates {
reply := "Не знаю что сказать"
msg := tgbotapi.NewMessage(update.Message.Chat.ID, reply)
msg.ReplyMarkup = board_StartCommand
if update.Message == nil {
continue
}
if _, err := bot.Send(msg); err != nil {
log.Panic(err)
}

}
}


Golang задачи с собеседований
👨🏻‍💻 Вопрос от пользователя

spf13/viper не может прочитать файл

По аналогии с https://github.com/gogjango/gjango/blob/master/config/config.go

package config

import (
"fmt"
"log"
"os"
"path/filepath"
"read-clickhouse/tools"
"runtime"

"github.com/gin-gonic/gin"
"github.com/spf13/viper"
)

func check() {
_, filePath, _, _ := runtime.Caller(0)
configName := "config.dev.yaml"
configPath := filePath[:len(filePath)-9] + "files" + string(filepath.Separator)

filename := fmt.Sprintf("%s%s", configPath, configName)

if _, err := os.Stat(filename); err == nil {
fmt.Println("Файл существует")
} else if os.IsNotExist(err) {
fmt.Println("Файл не существует")
} else {
fmt.Println("Ошибка при проверке файла:", err)
}
}

// Load returns Configuration struct
func Load(env string) *Configuration {
check()
_, filePath, _, _ := runtime.Caller(0)
configName := "config." + env + ".yaml"
configPath := filePath[:len(filePath)-9] + "files" + string(filepath.Separator)

tools.DebugPrintf(configName)
viper.SetConfigName(configName)
viper.AddConfigPath(configPath)

err := viper.ReadInConfig()
if err != nil {
log.Fatal(err)
}

var config Configuration
viper.Unmarshal(&config)
setGinMode(config.Server.Mode)

return &config
}

// Configuration holds data necessery for configuring application
type Configuration struct {
Server *Server `yaml:"server"`
}

// Server holds data necessary for server configuration
type Server struct {
Mode string `yaml:"mode"`
}

func setGinMode(mode string) {
switch mode {
case "release":
gin.SetMode(gin.ReleaseMode)
break
case "test":
gin.SetMode(gin.TestMode)
break
default:
gin.SetMode(gin.DebugMode)
}
}

вывод

go run main.go
Файл существует
[/home/des/proj/go-read-clickhouse/config/config.go:read-clickhouse/config.Load 38]
%!(EXTRA string=config.dev.yaml)

2024/02/02 16:34:51 Config File "config.dev.yaml" Not Found in "[/home/des/proj/go-read-clickhouse/config/files]"
exit status 1
make: *** [Makefile:2: run] Ошибка 1

сам файл в папке есть. ось - убунта. права 0775 накинул


Golang задачи с собеседований