ویرگول
ورودثبت نام
عرفان
عرفان
خواندن ۶ دقیقه·۶ سال پیش

حساب‌فان (دیتابیس و متعلقات)

حالا قصد داریم دیتابیس به پروژه اضافه کنیم، برای این کار به ORM و migration و seed نیاز داریم.

همونطور که تو پست مربوط به بک‌اند توضیح دادم از upper-db استفاده می‌کنم، چون خیلی سادس، داکیومنت خوبی داره، برای اکثر دیتابیس‌های ریلیشن، آداپتر داره و...

برای مایگریشن و سید از rubenv/sql-migrate استفاده میکنم چون فوق العاده سادس، استاندارده، چیز اضافه نداره و قابل اعتماده.

پس برای اضافه کردن پکیج دیتابیس فایل glide.yaml به این صورت تغییر میدیم:

package: gitlab.com/erfun/hesabFun homepage: https://hesabfun.com owners: - name: ErFUN KH email: erfun.kh@gmail.com homepage: https://erfun.net import: - package: github.com/gin-gonic/gin # testing package - package: github.com/stretchr/testify # ORM - package: github.com/go-sql-driver/mysql - package: upper.io/db.v3/mysql

و فایل main.go به این صورت تغییر میدیم:

package main import ( "upper.io/db.v3/mysql" "upper.io/db.v3/lib/sqlbuilder" "os" "log" ) var settings = mysql.ConnectionURL{ Database: "YOUR_MYSQL_DATABASE", Host: "YOUR_MYSQL_ADDRESS", User: "YOUR_MYSQL_USERNAME", Password: "YOUR_MYSQL_PASSWORD", } var MySql sqlbuilder.Database func main() { var DBError error MySql, DBError = mysql.Open(settings) if DBError != nil { log.Fatal("MySQL Error: ", DBError) } MySql.SetLogging(false) defer MySql.Close() router := setupRouter() router.Run() // listen and serve on 0.0.0.0:8080 }

خیلی خوب، الآن اپلیکیشن به دیتابیس MySQL وصل میشه، ولی همونطور که می‌بینید تمام اطلاعات دیتابیس از جمله یوزر و پسورد تو کد نوشته شده، قرار نیست همینطور بمونه چون اگر تیم داشته باشید ممکنه یوزر و پسورد دیتابیس هرکسی رو سیستم شخصیش فرق داشته باشه، همچنین دلیلی نداره یوزر و پسورد دیتابیس اصلی به همه دولوپرها بدید، از همه مهمتر برای تست به مشکل می‌خوریم چون قراره دیتابیس حسابی کثیف کنیم.

منم مثل همه از متغییرهای سیستم استفاده میکنم، با استفاده از پکیج goDotEnv یه فایل .env میسازیم و هردفعه اطلاعاتش میخونیم و رو سیستم ست میکنیم.

بنابراین پکیج به پکیج‌منیجر اضافه میکنیم:

package: gitlab.com/erfun/hesabFun homepage: https://hesabfun.com owners: - name: ErFUN KH email: erfun.kh@gmail.com homepage: https://erfun.net import: - package: github.com/gin-gonic/gin # testing package - package: github.com/stretchr/testify # ORM - package: github.com/go-sql-driver/mysql - package: upper.io/db.v3/mysql # Load environment variables from `.env` - package: github.com/joho/godotenv

همچنین جهت لود متغییرها پکیج goDotEnv لود میکنیم و تنظیمات دیتابیس کمی تغییر میدیم:

package main import ( "os" "log" "upper.io/db.v3/mysql" "upper.io/db.v3/lib/sqlbuilder" _ "github.com/joho/godotenv/autoload" ) var settings = mysql.ConnectionURL{ Database: os.Getenv("MYSQL_DATABASE"), Host: os.Getenv("MYSQL_ADDRESS"), User: os.Getenv("MYSQL_USERNAME"), Password: os.Getenv("MYSQL_PASSWORD"), } var MySql sqlbuilder.Database func main() { var DBError error MySql, DBError = mysql.Open(settings) if DBError != nil { log.Fatal("MySQL Error: ", DBError) } MySql.SetLogging(false) defer MySql.Close() router := setupRouter() router.Run() // listen and serve on 0.0.0.0:8080 }

الآن همه چی درسته ولی فایلی برای خوندن وجود نداره، باید فایل .env به پروژه اضافه کنیم ولی چون ممکنه کانفیگ هرکس تو سیستم خودش با دیگری فرق داشته باشه ما فایل به اسم .env.testing درست می‌کنیم و فایل .env تو گیت ایگنور میذاریم تاهرکس بعد کلون کردن پروژه فایل به .env تغییر نام بده و شخصی سازی کنه، پس به این صورت متغییرها تو فایل .env.testing معرفی می‌کنیم:

# MySQL Config MYSQL_DATABASE=hesab_fun MYSQL_ADDRESS=localhost MYSQL_USERNAME=root MYSQL_PASSWORD=123456 MYSQL_PORT=3306

همینطور فایل .gitignore به این صورت تغییر میدیم:

vendor .idea hesabFun .env

الآن رو سیستم MySQL نصب کنید به راحتی کانکت میشه البته یادتون نره دیتابیسی با اسم hesub_fun درست کنید، حالا باید بریم سراغ ساخت جدول‌های دیتابیس، برای این کار از مایگریشن استفاده میکنیم و هیچ‌وقت دستی دیتابیس دستکاری نمی‌کنیم.

من از مایگریشن در کامندلاین استفاده میکنم، ترجیح میدم پروژه خیلی سنگین نکنم و چیزایی که نیاز نیست بهش اضافه نکنم، اگه داکیومنتش خونده باشید متوجه می‌شید برای نصبش باید دستور go get -v github.com/rubenv/sql-migrate/... اجرا کنید.

حالا براش فایل کانفیگی با اسم dbconfig.yml ایجاد می‌کنیم:

development: dialect: mysql datasource: ${MYSQL_USERNAME}:${MYSQL_PASSWORD}@tcp(${MYSQL_ADDRESS}:${MYSQL_PORT})/$MYSQL_DATABASE?charset=utf8&parseTime=True&loc=Local dir: ./migrations table: migrations seed: dialect: mysql datasource: ${MYSQL_USERNAME}:${MYSQL_PASSWORD}@tcp(${MYSQL_ADDRESS}:${MYSQL_PORT})/$MYSQL_DATABASE?charset=utf8&parseTime=True&loc=Local dir: ./seeds table: seeds

میریم تو روت پروژه و فولدرهای migrations و seeds می‌سازیم، با دستور sql-migrate new create_users_table جدول کاربران ایجاد می‌کنیم.

حالا اگه بریم تو فولدر migrations فایل جدول کاربران می‌بینیم، محتوای فایل به همچین چیزی تغییر میدیم:

-- +migrate Up CREATE TABLE `users` ( `id` int(10) UNSIGNED NOT NULL, `created_at` timestamp NULL DEFAULT CURRENT_TIMESTAMP, `updated_at` timestamp NULL DEFAULT CURRENT_TIMESTAMP, `deleted_at` timestamp NULL DEFAULT null, `name` varchar(255) NOT NULL DEFAULT '', `email` varchar(255) NOT NULL DEFAULT '', `mobile` varchar(255) NOT NULL DEFAULT '', `password` varchar(255) DEFAULT NULL, `status` enum('pending','active','block') NOT NULL DEFAULT 'pending', `type` enum('user','admin','god') NOT NULL DEFAULT 'user', `remember_token` varchar(255) DEFAULT NULL, `sms_token` int(11) DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=utf8; ALTER TABLE `users` ADD PRIMARY KEY (`id`), ADD UNIQUE KEY `email` (`email`,`mobile`); ALTER TABLE `users` MODIFY `id` int(10) UNSIGNED NOT NULL AUTO_INCREMENT; COMMIT; -- +migrate Down DROP TABLE `users`;

حالا با دستور sql-migrate new -env=seed create_user فایل جدیدی برای ساخت یوزر تست ایجاد می‌کنیم با همچین محتوایی:

-- +migrate Up INSERT INTO `users` ( `created_at`, `updated_at`, `deleted_at`, `name`, `email`, `mobile`, `password`, `status`, `type`, `remember_token`, `sms_token`) VALUES ('2018-02-04 18:10:59', '2018-02-04 18:10:59', null, 'عرفان', '', '09111111111', '25d55ad283aa400af464c76d713c07ad', 'active', 'user', '', 0); -- +migrate Down DELETE FROM `users` WHERE `users`.`mobile` = '09111111111';

خب حالا میخوایم میخوایم جدول ساخته بشه و دیتای تست بهش وارد بشه، قبل هرچیزی باید متغییرهای دیتابیس به سیستم مون اضافه کنیم، تو مک به این صورت عمل میکنیم. اول فایل .bash_profile باز میکنیم:

nano ~/.bash_profile

بعد محتوای زیر بیش اضافه می‌کنیم:

export MYSQL_DATABASE=hesab_fun export MYSQL_ADDRESS=localhost export MYSQL_USERNAME=root export MYSQL_PASSWORD= export MYSQL_PORT=3306

توجه کنید اگر تنظیمات دیتابیس شما غیر اینه، حتما تو فایل بالا تغییرش بدید.

حالا کافیه تو روت پروژه دستور sql-migrate up بزنیم تا جدول users ایجاد بشه، برای اضافه کردن دیتای تست دستور sql-migrate up -env=seed میزنیم و دیتا به جدول کاربران اضافه میشه.

تا اینجا همه چی بی نقصه، حالا باید بریم سراغ تست اپ، فایل .gitlab-ci.yml باز میکنیم و به این صورت تغییرش میدیم:

image: golang:1.10 stages: - build - test services: - mysql:latest variables: MYSQL_DATABASE: hesab_fun MYSQL_ROOT_PASSWORD: 123456 MYSQL_ADDRESS: mysql MYSQL_PORT: 3306 MYSQL_PASSWORD: 123456 MYSQL_USERNAME: root before_script: - curl https://glide.sh/get | sh - mkdir -p /go/src/gitlab.com/ - cp -r /builds/erfun /go/src/gitlab.com/erfun - cd /go/src/gitlab.com/erfun/hesabFun - glide install - go get github.com/rubenv/sql-migrate/... - sql-migrate up - sql-migrate up -env=seed build-project: stage: build script: - go build test-project: stage: test script: - go test -v -cover

اول از همه سرویس MySQL اضافه کردیم، بعد متغییرها دیتابیس اضافه کردیم، در آخر هم مایگریشن نصب و اجرا کردیم.

اگه به نظرتون پیچیده شد، توصیه میکنیم کامیت‌ها تو گیت لب چک کنید و با دقت بخونید. بعد هر پست کدها کامیت می‌کنم.

upperdatabasemysqlgolangmigration
یه دولوپر که سعی می‌کنه عمیق و کم هزینه باشه، از هرچیزی که بلدم می‌نویسم تا مطمئن‌شم درست یادش گرفتم.
شاید از این پست‌ها خوشتان بیاید