Представим, что хочется, чтобы котики чувствовали себя свободно, но и их хозяева не переживали. Сделаем сервис с отображением котиков на карте.
Данные будем хранить и индексировать в Tarantool Обрабатывать запросы будет в golang Рисовать карту на html/js
Вот что нам нужно сделать:
- Фронтенд на HTML/JS с Leaflet и OpenStreetMap
- Виджет с картой
- События пользователя
- Запросы к Golang-бекенду
- Создание Golang приложения
- Подключение к базе
- Индексирование
- Запросы к базе
- HTTP-сервер
- HTTP API
- Конфигурация базы данных
Начнем с html
Сделаем файл index.html
touch index.html
<html>
</html>
Подключим фреймворк для рисования карты — leaflet
<head>
<title>The Map</title>
<link rel="stylesheet" href="https://unpkg.com/[email protected]/dist/leaflet.css" crossorigin="" />
<script src="https://unpkg.com/[email protected]/dist/leaflet.js" crossorigin=""></script>
<script src="https://unpkg.com/[email protected]/leaflet-providers.js" crossorigin=""></script>
</head>
Нарисуем виджет с картой
<body>
<!-- div для карты -->
<div id="mapid" style="height:100%"></div>
<script>
// Карта
var mymap = L.map('mapid',
{ 'tap': false })
.setView([59.95, 30.31], 13)
// Слой карты с домами, улицами и т.п.
var osm = L.tileLayer('https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png', {
maxZoom: 19,
attribution: '© <a href="https://www.openstreetmap.org/copyright">OpenStreetMap</a> contributors',
}).addTo(mymap)
</script>
</body>
Добавим слой и добавим на него тестовый маркер просто чтобы убедится
group = L.layerGroup().addTo(mymap)
L.marker([59.95, 30.31]).addTo(group).bindPopup("Hello World")
Должно всё получиться
Создадим функцию, которая будет добавлять объекты на карту по клику мыши
// Обрабатываем нажатие на карту
function onMapClick(e) {
L.marker(e.latlng).addTo(group).bindPopup("Hello World")
}
mymap.on('click', onMapClick)
- Redis говорит: «Сконфигурируй меня с помощью простых команд в файле»
- Mongo говорит: «Сконфигурируй меня с помощью YAML файла»
- Tarantool говорит: «Используй Lua скрипт»
Чтобы запустить Tarantool и подключиться к нему, нам понадобится всего три Lua-функции:
box.cfg
box.schema.user.create
box.schema.user.grant
box.cfg
Функция настраивает весь Tarantool. Часть параметров можно задать только один раз при старте. Другую часть можно менять в любой момент времени.
box.schema.user.create
Функция создаёт пользователя для удаленной работы.
box.schema.user.grant
Функция перечисляет, что можно и что нельзя будет делать пользователю.
Конфигурация Tarantool-а — единственное место, где будет Lua.
Создадим файл init.lua
touch init.lua
init.lua
-- Открываем порт для доступа по iproto
box.cfg({listen="127.0.0.1:3301"})
-- Создаём пользователя для подключения
box.schema.user.create('storage', {password='passw0rd', if_not_exists=true})
-- Даём все-все права
box.schema.user.grant('storage', 'super', nil, nil, {if_not_exists=true})
На одном узле Tarantool находится только одна база данных. Данные складываются в спейсы == таблицы в мире SQL. К данным обязательно строится первичный индекс, а количество вторичных произвольно.
Для хранения маркеров сделаем таблицу:
id | coordinates | name |
---|---|---|
string | [double, double] | string |
В поле id
хранится уникальный идентификатор, который мы сами сгенерируем.
В поле coordinates
— координаты маркера (массив из двух double).
В поле name
— строка с кличкой.
-- создаём таблицу для хранения отзывов на карте
box.schema.space.create('cats', {if_not_exists=true})
box.space.cats:format({
{name="id", type="string"},
{name="coordinates", type="array"},
{name="name", type="string"}
})
-- создаём первичный индекс
box.space.cats:create_index('primary', {
parts={{ field="id", type="string" }},
type = 'TREE',
if_not_exists=true,})
-- создаём индекс для координат
box.space.cats:create_index('spatial', {
parts = {{ field="coordinates", type='array'} },
type = 'RTREE',
unique = false,
if_not_exists=true,})
Запущу локальную консоль внутри базы данных, для всяких тестовых команд
require('console').start() os.exit()
На этом Lua закачивается.
tarantool init.lua
Создадим файл cats.go
touch cats.go
package main
import (
"encoding/json"
"net/http"
// Генерация уникальных идентификаторов
"github.com/chilts/sid"
// Коннектор к Tarantool
"github.com/tarantool/go-tarantool"
)
// Структура для сериализации гео объектов в/из Tarantool
type CatObject struct {
Id string `json:"id"`
Coordinates [2]float64 `json:"coordinates"`
Name string `json:"name"`
}
func main() {
opts := tarantool.Opts{User: "storage", Pass: "passw0rd"}
conn, err := tarantool.Connect("127.0.0.1:3301", opts)
if err != nil {
panic(err)
}
defer conn.Close()
}
Запуск
go run ./cats.go
go mod init cats
go get github.com/tarantool/go-tarantool
go run ./cats.go
Добавим хостинг index.html
файла
// В корневом эндпоинте отдаём пользователю фронтенд
http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
http.ServeFile(w, r, "index.html")
})
// Запускаем http сервер на локальном адресе
err = http.ListenAndServe("127.0.0.1:8080", nil)
if err != nil {
panic(err)
}
// Эндпоинт для сохранения маркера
http.HandleFunc("/put", func(w http.ResponseWriter, r *http.Request) {
dec := json.NewDecoder(r.Body)
obj := &CatObject{}
err := dec.Decode(obj)
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
// Генерируем уникальный идентификатор маркера
if obj.Id == "" {
obj.Id = sid.IdHex()
}
var tuples []CatObject
// Вставляем новый маркер
err = conn.ReplaceTyped("cats", []interface{}{obj.Id, obj.Coordinates, obj.Name}, &tuples)
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
enc := json.NewEncoder(w)
enc.Encode(tuples)
r.Body.Close()
})
go get github.com/chilts/sid
go run ./cats.go
function onMapClick(e) {
var name = window.prompt('Cat name?')
if (name != null) {
L.marker(e.latlng).addTo(group).bindPopup(name)
/*
* Карта использует систему координат на шаре
* Tarantool хранит координаты на плоскости
* Конвертируем из одной системы в другую
*/
var p = mymap.project(e.latlng, 1)
var cat = {
"coordinates": [p.x, p.y],
"name": name,
}
fetch("/put", {
method: "POST",
body: JSON.stringify(cat)
})
}
}
В html добавим функцию для загрузки котиков в рамках экрана
function addCat(cat) {
var l = mymap.unproject(L.point(cat['coordinates']), 1)
var name = cat['name']
// Создаем маркер
L.marker(l).addTo(group).bindPopup(name)
}
// Обрабатываем json пришедший с сервера
function parse(array) {
array.forEach(addCat)
}
function errorResponse(error) {
alert('Error: ' + error)
}
function handleListResponse(res) {
res.json().then(parse).catch(errorResponse)
}
function onMapMove(e) {
var bounds = mymap.getBounds()
var northeast = bounds.getNorthEast()
var southwest = bounds.getSouthWest()
var ne = mymap.project(northeast, 1)
var sw = mymap.project(southwest, 1)
var options = {
"rect": JSON.stringify([ne.x, ne.y, sw.x, sw.y]),
}
// Отправляем запрос на сервер с получением маркеров
fetch("/list?" + new URLSearchParams(options))
.then(handleListResponse)
.catch(errorResponse)
}
mymap.on('move', onMapMove)
onMapMove()
// Отдаём маркеры для указанного в url региона
http.HandleFunc("/list", func(w http.ResponseWriter, r *http.Request) {
rect, ok := r.URL.Query()["rect"]
if !ok || len(rect) < 1 {
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
var arr []float64
err := json.Unmarshal([]byte(rect[0]), &arr)
if err != nil {
panic(err)
}
// Запрашивает 1000 маркеров, которые находятся в регионе rect
var tuples []CatObject
err = conn.SelectTyped("cats", "spatial", 0, 1000, tarantool.IterLe,
arr,
&tuples)
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
enc := json.NewEncoder(w)
enc.Encode(tuples)
r.Body.Close()
})
Добавим файл с котиком
- index.html
var icon = L.icon({
iconUrl: "",
iconSize: [32, 32],
iconAnchor: [0, 0],
popupAnchor: [0, 0],
})
L.marker(l, {"icon": icon}).addTo(group).bindPopup(name)
Маркеры на карте создаются. При навигации маркеры загружаются с бека.
При навигации маркеры постоянно добавиляются, даже если уже были.
Решение
Выполнять проверку на фронтенде, если маркер уже был отрисован.
var alreadyloaded = {}
function addCat(cat) {
if (!(cat.id in alreadyloaded)) {
var l = mymap.unproject(L.point(cat['coordinates']), 1)
var name = cat['name']
// Создаем маркер
L.marker(l, {"icon": icon}).addTo(group).bindPopup(name)
alreadyloaded[cat.id] = cat
}
}
Сделаем приложение бота, которое будет рассказывать нам о передвижениях котов. В реальном мире это мог бы быть аггрегатор gps координат с ошейников наших питомцев.
Файл catpaths.go
touch catpaths.go
catpaths.go
package main
import (
"bytes"
"encoding/json"
"io"
"io/ioutil"
"math/rand"
"net/http"
"sync"
"time"
"github.com/chilts/sid"
"github.com/tarantool/go-tarantool"
)
var names = []string{
"Боня",
"Ася",
"Алиса",
"Багира",
"Бусинка",
"Буся",
"Аврора",
"Муся",
"Агния",
"Ева",
"Мася",
"Агата",
"Василиса",
"Соня",
"Агаша",
"Мурка",
"Муська",
"Нюша",
"Бася",
"Симка",
"Абракадабра",
"Ангел",
"Багирка",
"Аба",
"Анабель",
"Абби",
"Сима",
"Ванесса",
"Адель",
"Дымка",
"Абигель",
"Бакси",
"Барселона",
"Масяня",
"Абалина",
"Даша",
"Гера",
"Агнесса",
"Альфа",
"Бэлла",
"Амели",
"Джессика",
"Айса",
"Барса",
"Карамелька",
"Бан-Ши",
"Джесси",
"Ириска",
"Китти",
"Агнес",
"Айрис",
"Кака",
"Барсик",
"Боня",
"Бакс",
"Алекс",
"Бади",
"Амур",
"Ебони",
"Абуссель",
"Баксик",
"Жопкинс",
"Кузя",
"Персик",
"Абрек",
"Абрикос",
"Тимоша",
"Авалон",
"Бабник",
"Саймон",
"Бурбузяка",
"Абу",
"Марсик",
"Маркиз",
"Дымок",
"Лаки",
"Симба",
"Абрамович",
"Сёма",
"Пушок",
"Айс",
"Бося",
"Алмаз",
"Кекс",
"Басик",
"Макс",
"Феликс",
"Гарфилд",
"Том",
"Тиша",
"Цезарь",
"Тишка",
"Мася",
"Абакан",
"Лакки",
"Васька",
"Адольф",
"Марсель",
"Бабасик",
"Вася",
"Зевс",
"Вольт",
"Адидас",
"Лео",
}
func randName() string {
return names[rand.Intn(len(names))]
}
func randFloat(min float64, max float64) float64 {
return min + rand.Float64()*(max-min)
}
func main() {
opts := tarantool.Opts{User: "storage", Pass: "passw0rd"}
conn, err := tarantool.Connect("127.0.0.1:3301", opts)
if err != nil {
panic(err)
}
defer conn.Close()
conn.Call("box.space.geo:truncate", []interface{}{})
tr := &http.Transport{
MaxConnsPerHost: 50,
}
defer tr.CloseIdleConnections()
cl := &http.Client{
Transport: tr,
}
// Питерские координаты
bounds := []float64{298.02, 148.52, 299.64, 149.20}
centerx := 299.12
centery := 148.80
rand.Seed(time.Now().UnixNano())
data := make(map[string]map[string]interface{})
for i := 0; i < 1e3; i++ {
id := sid.IdHex()
item := map[string]interface{}{
"id": id,
"coordinates": []float64{randFloat(bounds[0], bounds[2]), randFloat(bounds[1], bounds[3])},
"name": randName(),
}
data[id] = item
bytes := new(bytes.Buffer)
json.NewEncoder(bytes).Encode(item)
resp, err := cl.Post("http://127.0.0.1:8080/put", "application/json", bytes)
if err != nil {
panic(err)
}
resp.Body.Close()
}
var wg sync.WaitGroup
parralel := 0
for step := 0; step < 1e6; step++ {
for _, source := range data {
value := make(map[string]interface{})
value["coordinates"] = source["coordinates"]
value["id"] = source["id"]
value["name"] = source["name"]
coords := value["coordinates"].([]float64)
parralel = parralel + 1
wg.Add(1)
go func() {
if rand.Int31n(4) == 0 {
coords[0] = coords[0] + 0.0003
} else {
coords[0] = coords[0] - 0.0003
}
if rand.Int31n(4) == 0 {
coords[1] = coords[1] + 0.0003
} else {
coords[1] = coords[1] - 0.0003
}
if coords[0] > centerx {
coords[0] = coords[0] - 0.0003
} else {
coords[0] = coords[0] + 0.0003
}
if coords[1] > centery {
coords[1] = coords[1] - 0.0003
} else {
coords[1] = coords[1] + 0.0003
}
if coords[0] < bounds[0] {
coords[0] = bounds[0]
}
if coords[0] > bounds[2] {
coords[0] = bounds[2]
}
if coords[1] < bounds[1] {
coords[1] = bounds[1]
}
if coords[1] > bounds[3] {
coords[1] = bounds[3]
}
value["coordinates"] = coords
bytes := new(bytes.Buffer)
json.NewEncoder(bytes).Encode(value)
resp, err := cl.Post("http://127.0.0.1:8080/put", "application/json", bytes)
if err != nil {
panic(err)
}
io.Copy(ioutil.Discard, resp.Body)
resp.Body.Close()
wg.Done()
}()
if parralel == 1 {
parralel = 0
wg.Wait()
}
}
time.Sleep(10 * time.Millisecond)
}
}
Запуск
go run ./cat_paths.go
Теперь если обновлять страницу, то маркеры будут двигаться.
Добавим автоматические обновление страницы.
redraw = setInterval(onMapMove, 1)
Сделаем изменение координат маркеров.
var alreadyloaded = {}
var popups = {}
function addCat(cat) {
if (!(cat.id in alreadyloaded)) {
var l = mymap.unproject(L.point(cat['coordinates']), 1)
var name = cat['name']
// Создаем маркер
popups[cat.id] = L.marker(l, {"icon": icon}).addTo(group).bindPopup(name)
alreadyloaded[cat.id] = cat
} else {
var l = mymap.unproject(L.point(cat['coordinates']), 1)
popups[cat.id].setLatLng(l)
}
}
Сделаем файл cats_test.go
c тестами и проверим
touch cats_test.go
package main
import (
"bytes"
"encoding/json"
"io/ioutil"
"math/rand"
"net/http"
"testing"
)
const letterBytes = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"
func RandStringBytes(n int) string {
b := make([]byte, n)
for i := range b {
b[i] = letterBytes[rand.Intn(len(letterBytes))]
}
return string(b)
}
func BenchmarkWrite(b *testing.B) {
tr := &http.Transport{}
defer tr.CloseIdleConnections()
cl := &http.Client{
Transport: tr,
}
var x = rand.Float64()
var y = rand.Float64()
var data = map[string]interface{}{
"coordinates": []float64{x, y},
"name": RandStringBytes(100),
}
for i := 0; i < b.N; i++ {
data["coordinates"] = []float64{float64(rand.Int31n(1000)) + rand.Float64(), float64(rand.Int31n(1000)) + rand.Float64()}
bytes := new(bytes.Buffer)
json.NewEncoder(bytes).Encode(data)
res, err := cl.Post("http://127.0.0.1:8080/put", "application/json", bytes)
if err != nil {
b.Fatal("Post:", err)
}
_, err = ioutil.ReadAll(res.Body)
if err != nil {
b.Fatal("ReadAll:", err)
}
}
}
func BenchmarkRead(b *testing.B) {
tr := &http.Transport{}
defer tr.CloseIdleConnections()
cl := &http.Client{
Transport: tr,
}
for i := 0; i < b.N; i++ {
bytes, err := json.Marshal([]float64{float64(rand.Int31n(1000)) + rand.Float64(),
float64(rand.Int31n(1000)) + rand.Float64(),
float64(rand.Int31n(360)) + rand.Float64(),
float64(rand.Int31n(360)) + rand.Float64()})
if err != nil {
b.Fatal(err)
}
res, err := cl.Get("http://127.0.0.1:8080/list?rect=" + string(bytes))
if err != nil {
b.Fatal(err)
}
var objects []CatObject
dec := json.NewDecoder(res.Body)
err = dec.Decode(&objects)
if err != nil {
b.Fatal(err)
}
res.Body.Close()
}
}
Запуск тестов
go test -benchmem -benchtime 10s -bench BenchmarkWrite
Подключим реплику на чтение В golang писать будем в мастер, читать будем с реплики
replica.lua
touch replica.lua
box.cfg{work_dir="replica", listen=3302, replication="storage:[email protected]:3301"}
Создадим рабочую директорию для реплики
mkdir replica
Запустим реплику
tarantool replica.lua
Добавим в файл cats.go подключение к readonly реплике
opts = tarantool.Opts{User: "storage", Pass: "passw0rd"}
readconn, err := tarantool.Connect("127.0.0.1:3302", opts)
if err != nil {
panic(err)
}
defer readconn.Close()
err = readconn.SelectTyped("cats", "spatial", 0, 1000, tarantool.IterLe,
arr,
&tuples)
Запустим котиков. Потому запустим нагрузочные тесты.
Пользуйтесь Тарантулом, он данные хранит и вам быстро отдаёт. Tarantool это масштабируемый OLTP. Clickhouse это масштабируемый OLAP.
Можно писать микросервисы на Tarantool рядом с данными.
Что осталось за кадром:
- Шардирование
- Автоматический фейловер
- Синхронная репликация
Вопросы?