021-88881776

آموزش ماژول‌ها و ابزارهای Go

در این مقاله با آموزش go آشنا می‌شویم و قدم به دنیای جذاب ماژول‌ها و ابزارهای Go می‌گذاریم. زبان Go به دلیل سادگی در نحو، سرعت بالای اجرا و امکانات ماژولار قدرتمند، توجه بسیاری از برنامه‌نویسان را به خود جلب کرده است. اگر به تازگی کار با این زبان را آغاز کرده‌اید یا قصد دارید دانش خود را در زمینه مدیریت وابستگی، انتشار ماژول، تست و بهینه‌سازی کد در Go گسترش دهید، این مقاله بهترین نقطه شروع برای شما خواهد بود. در ادامه، هر یک از مفاهیم اساسی ماژول‌ها و ابزارهای Go را با مثال‌های کاربردی بررسی کرده و تلاش می‌کنیم مسیری ساده و ساختارمند برای یادگیری عمیق این ابزارها در اختیارتان قرار دهیم.

ماژول‌ها و ابزارهای Go

در توسعه نرم‌افزارهای مبتنی بر زبان Go، شناخت ماژول‌ها و ابزارهای Go از اهمیت ویژه‌ای برخوردار است. این ماژول‌ها در واقع همان بسته‌های مستقل یا پکیج‌های بزرگ‌تری هستند که مدیریت وابستگی‌ها و نسخه‌بندی را برای ما ساده می‌کنند. به‌ عبارت دیگر، وقتی یک پروژه بزرگ Go توسعه می‌دهید، ممکن است نیاز داشته باشید تا کتابخانه‌ها یا بسته‌هایی را از منابع خارجی ‘third−party‘ دریافت کنید. سیستم ماژول در Go، تمامی این وابستگی‌ها را گردآوری و نسخه‌بندی می‌کند تا مطمئن شوید پروژه شما با یک نسخه مشخص از آن کتابخانه‌ها کار می‌کند و در صورت به‌روزرسانی آن کتابخانه‌ها، پروژه‌تان دچار مشکل نشود.
علاوه بر این، ماژول‌ها و ابزارهای Go فقط به مدیریت وابستگی خلاصه نمی‌شوند؛ ابزارهای متنوع Go مانند دستورات go build (برای ساخت باینری و پکیج نهایی)، go run (برای اجرای کد به‌صورت سریع و مستقیم)، go test (برای تست خودکار بخش‌های مختلف کد) و go fmt (برای فرمت‌بندی استاندارد کد) در کنار ماژول‌ها، بستری کامل و قدرتمند برای توسعه نرم‌افزار فراهم می‌کنند. به کمک این ابزارها می‌توانید:

سازماندهی کد و ساختار پروژه: از آنجا که ماژول‌ها یک ساختار مشخص ‘go.mod‘،‘go.sum‘ برای ذخیره اطلاعات پروژه فراهم می‌کنند، می‌توانید پروژه‌های خود را در سطح یک سازمان یا حتی در گیت‌های عمومی مانند GitHub و GitLab به‌راحتی مدیریت کنید.

کنترل نسخه و تغییرات: سیستم ماژول به شما این امکان را می‌دهد که نسخه‌های مختلف از کتابخانه‌های خارجی یا ماژول‌های داخلی را تحت کنترل بگیرید؛ می‌توانید نسخه مشخصی را قفل کنید و مطمئن شوید تا زمانی که خودتان نخواهید، آن کتابخانه به نسخه جدیدتر ارتقا پیدا نمی‌کند.

ساخت و اجرا: ابزار go build به شما اجازه می‌دهد سورس‌کد خود را کامپایل کرده و خروجی باینری دریافت کنید. همچنین go run این امکان را به شما می‌دهد که بدون نیاز به ساخت باینری مجزا، کد را در لحظه اجرا کنید. این دو دستور باعث تسهیل فرآیند توسعه و اشکال‌زدایی می‌شوند.

تست خودکار: به کمک go test می‌توانید توابع و بخش‌های مختلف برنامه خود را به‌صورت خودکار آزمایش کنید. این تست‌ها به شما اطمینان می‌دهند که پس از هر تغییر در کد، عملکرد برنامه کماکان درست است و باگ جدیدی وارد پروژه نشده است.

فرمت‌بندی کد و رعایت استانداردها: ابزار go fmt به‌صورت پیش‌فرض در دسترس است و تمامی فایل‌های .go پروژه را بر اساس استانداردهای تعیین‌شده فرمت می‌کند. نتیجه این کار، یکپارچگی در سبک کدنویسی تیمی و تسهیل در خوانایی کد برای دیگر توسعه‌دهندگان است.

در مجموع، وقتی صحبت از ماژول‌ها و ابزارهای Go می‌شود، منظور مجموعه‌ای از قابلیت‌های داخلی زبان Go است که به توسعه‌دهندگان کمک می‌کند پروژه‌های خود را ساده‌تر، استانداردتر و در نتیجه با کیفیت بالاتر پیش ببرند. این ویژگی‌ها باعث می‌شود تا علاوه بر کدنویسی روان، در زمان و منابع نیز صرفه‌جویی شود و تیم‌ها بتوانند به‌صورت همزمان و بدون تداخل در پروژه‌ها کار کنند. در ادامه مقاله، با جزئیات بیشتری به هریک از این ابزارها و شیوه‌های استفاده مؤثر از سیستم ماژول در Go خواهیم پرداخت تا مسیر آموزش go برای شما هموارتر و جذاب‌تر شود.

تعریف ماژول‌ها

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

در گذشته، پیش از معرفی سیستم ماژول در نسخه 1.11، مدیریت وابستگی‌ها به شکل دستی یا توسط ابزارهایی مانند dep صورت می‌گرفت. این روش‌ها در پروژه‌های کوچک یا متوسط تا حدی قابل استفاده بودند، اما برای پروژه‌های بزرگ چالش‌هایی نظیر ناسازگاری نسخه‌ها، دشواری در به‌روزرسانی وابستگی‌ها و عدم اطمینان از امنیت کتابخانه‌ها را به همراه داشتند. با آمدن ماژول‌ها و ابزارهای Go، این مشکلات به شکل استاندارد و رسمی برطرف شد و فرایند نسخه‌بندی و مدیریت وابستگی‌ها تسهیل گردید.

ساختار یک ماژول چگونه است؟

هر ماژول با فایل کلیدی go.mod شناسایی می‌شود. این فایل هویت ماژول و اطلاعات مرتبط با آن را در بر دارد. می‌توان گفت go.mod شناسنامه ماژول شما است و اطلاعاتی همچون نسخه زبان Go، نام ماژول و کتابخانه‌هایی که به آن‌ها وابسته هستید را ذخیره می‌کند. از سویی دیگر، فایل go.sum حکم مدرک امنیتی را دارد که هش و امضای هر وابستگی را نگهداری می‌کند. این کار برای اطمینان از صحت کتابخانه‌هایی که دانلود می‌کنید، ضروری است.

اگر ساختار کلی یک ماژول را در قالب درختی نگاه کنیم، ممکن است شبیه زیر باشد:

my-module/
  ├── go.mod
  ├── go.sum
  ├── main.go
  ├── utils/
  │   └── helper.go
  └── ...

go.mod:

نام ماژول را مشخص می‌کند (برای مثال module github.com/username/my-module)
حاوی نسخه زبان Go است (مانند go 1.19)
در بخش وابستگی‌ها، نام پکیج‌ها و ورژن یا هش آن‌ها را ذکر می‌کند.
go.sum:

آرشیوی از هش‌ها و امضاهای کتابخانه‌ها است تا از صحت و اعتبار آن‌ها اطمینان حاصل شود.
این فایل به‌صورت خودکار توسط ابزارهای Go تولید و به‌روزرسانی می‌شود و معمولاً نیازی به دستکاری دستی آن نیست.
main.go:

اگر ماژول شما در حالت برنامه اجرایی ‘executable‘ باشد، فایل main.go نقش ورودی ‘entrypoint‘ را بر عهده می‌گیرد.
در این فایل معمولاً پکیج main تعریف می‌شود و تابع main() نقطه شروع اجرای برنامه است.

پوشه‌ها و پکیج‌های جانبی (نظیر utils/helper.go):

می‌توانید در پوشه‌هایی مانند utils، پکیج‌های جداگانه ایجاد کنید تا منطق کد از فایل اصلی مجزا باشد.
هر پکیج توابع و متدهای مختص به خود را دارد و از طریق ایمپورت ‘import”my−module/utils”‘ در بخش‌های دیگر پروژه قابل استفاده است.

چرا ماژول‌ها مهم هستند؟

مدیریت وابستگی‌ها: مهم‌ترین مزیت استفاده از ماژول‌ها، مدیریت هوشمند وابستگی‌ها در پروژه است. تنها با اجرای چند دستور ساده مثل go mod tidy، می‌توانید وابستگی‌های اضافی را حذف و کتابخانه‌های مورد نیاز را اضافه کنید.
کنترل نسخه‌: امکان قفل کردن ‘pin‘ نسخه خاصی از هر کتابخانه، باعث می‌شود پروژه شما همواره با نسخه‌ای تست‌شده و مطمئن کار کند و در برابر تغییرات ناگهانی مقاوم باشد.
انتشار آسان: وقتی تصمیم می‌گیرید کد یا کتابخانه خود را برای دیگران نیز قابل استفاده کنید، کافیست آن را در یک مخزن ‘repository‘ عمومی مانند GitHub قرار داده و نام ماژول در go.mod را به آدرس مخزن اختصاصی خود تنظیم کنید. سپس دیگران می‌توانند به‌راحتی ماژول شما را با go get نصب کنند.
سازماندهی بهتر: کدها را می‌توان در پکیج‌ها و ماژول‌های مجزا دسته‌بندی کرد و این امر باعث می‌شود ساختار پروژه تمیزتر، ماژولارتر و قابل نگهداری‌تر باشد.
در مجموع، با معرفی سیستم ماژول در Go، فرایند توسعه پروژه‌های مقیاس‌پذیر و حرفه‌ای بسیار ساده‌تر شده است. اکنون توسعه‌دهندگان می‌توانند با اطمینان از نسخه‌های کتابخانه‌های مورد استفاده، روی منطق اصلی برنامه متمرکز شوند. این نکته در کنار سایر ویژگی‌های ماژول‌ها و ابزارهای Go باعث شده است تا Go در بسیاری از پروژه‌های بزرگ و کوچک، انتخاب محبوبی برای توسعه‌دهندگان باشد.

مدیریت وابستگی‌ها با go mod

مدیریت وابستگی‌ها با go mod یکی از ارکان مهم اکوسیستم ماژول‌ها و ابزارهای Go است که باعث می‌شود فرآیند افزودن، حذف و به‌روزرسانی کتابخانه‌های خارجی ‘third−partylibraries‘ در پروژه‌ها بسیار ساده‌تر شود. در ادامه، توضیحات جامع‌تری در مورد نحوه عملکرد go mod، ساختار فایل‌های ماژول و ترفندهای پیشرفته برای کنترل بهتر نسخه‌ها ارائه می‌دهیم.

چرا go mod معرفی شد؟

پیش از نسخه 1.11، مدیریت وابستگی‌ها در Go به روش‌های گوناگونی مانند استفاده از ابزارهای جانبی (نظیر dep) یا حتی به صورت دستی انجام می‌شد. این روش‌ها در پروژه‌های کوچک قابل قبول بودند، اما در پروژه‌های بزرگ، مشکلات جدی ناسازگاری نسخه و سردرگمی در به‌روزرسانی کتابخانه‌ها رخ می‌داد. با ورود go mod:

استانداردسازی: از نسخه 1.11 به بعد، مدیریت ماژول‌ها و وابستگی‌ها در دل زبان Go نهادینه شد و دیگر نیاز به ابزارهای خارجی نداشتید.
سازماندهی شفاف: فایل‌های go.mod و go.sum ساختار و نسخه‌های کتابخانه‌های پروژه را به وضوح مشخص می‌کنند.
بهبود همزمانی ‘concurrency‘: با توجه به فلسفه Go در تسهیل برنامه‌نویسی همزمان، مدیریت بهینه وابستگی‌ها بر پایداری و یکپارچگی کد کمک شایانی می‌کند.

اجزای اصلی فایل go.mod

فایل go.mod علاوه بر تعریف نام ماژول و نسخه زبان Go، مجموعه‌ای از دستورات ‘directives‘ مهم را در بر دارد که به صورت خودکار یا دستی ویرایش می‌شوند:

require: بیانگر وابستگی پروژه به یک کتابخانه خاص در نسخه مشخص است. مثلاً:

require (
    github.com/gorilla/mux v1.8.0
    golang.org/x/sync v0.1.0
)

replace: اگر قصد دارید یک وابستگی را با نسخه دیگری از همان پکیج یا حتی با یک مخزن محلی جایگزین کنید، از replace استفاده می‌کنید. برای نمونه:

replace github.com/gorilla/mux => github.com/gorilla/mux v1.7.3

این امکان برای زمانی مفید است که بخواهید نسخه خاصی از یک کتابخانه یا یک فورک
‘fork‘ سفارشی را در پروژه خود استفاده کنید.

exclude: برای حذف نسخه خاصی از یک کتابخانه از محدوده‌ی نصب
‘dependencygraph‘ به کار می‌رود. به طور مثال:

exclude github.com/gorilla/mux v1.9.0

با این دستور، اگر کتابخانه‌های دیگر به نسخه 1.9.0 از mux اشاره کنند، از آن نسخه صرف‌نظر می‌شود.

دستور go get در حالت‌های مختلف
نصب وابستگی جدید:
با اجرای

go get github.com/gorilla/mux

نسخه‌ای سازگار با نسخه‌ی زبان Go و سایر وابستگی‌های پروژه برای mux انتخاب شده و فایل‌های go.mod و go.sum به‌روزرسانی می‌شوند.

به‌روزرسانی وابستگی موجود:

go get -u github.com/gorilla/mux

نسخه جدیدتر کتابخانه بارگیری شده و مجدداً go.mod و go.sum اصلاح می‌شوند. اگر نسخه جدید با نسخه قبلی در تضاد باشد، ممکن است نیاز به بررسی یا اصلاح کد داشته باشید.

نصب ابزار اجرایی ‘binary‘:
در نسخه‌های جدید Go (از 1.17 به بعد)، برای نصب ابزارهای خارجی معمولاً از go install استفاده می‌شود. اما همچنان می‌توانید از go get نیز بهره ببرید. برای مثال:

go install github.com/pressly/goose/v3@latest

این دستور، نسخه آخر ابزار goose را دانلود و در $GOPATH/bin یا محل تعیین شده نصب می‌کند (بسته به پیکربندی Go).

دستور go mod tidy: کلید تمیزکاری فایل‌های ماژول

پس از حذف یا افزودن کتابخانه‌ها، ممکن است کد اصلی پروژه دیگر از برخی وابستگی‌ها استفاده نکند. دستور go mod tidy برای رفع این مشکل طراحی شده است:

go mod tidy

وابستگی‌های بی‌استفاده را از go.mod حذف می‌کند.
نسخه‌ها و هش‌های مربوط به کتابخانه‌های فعال را در go.sum به‌روزرسانی کرده و هر مورد اضافی را پاک می‌کند.
به حفظ سلامت و حداقل بودن ‘minimal‘ حجم وابستگی‌ها کمک شایانی می‌کند.

مدیریت وابستگی‌های لوکال با replace

گاهی اوقات می‌خواهید کتابخانه‌ای را که در لوکال ‘localrepository‘ خود دارید، آزمایش یا ویرایش کنید بدون اینکه آن را بلافاصله در مخزن اصلی
‘remote‘ آپلود کنید. برای این منظور، در go.mod می‌توانید از دستور replace بهره بگیرید. نمونه‌ای از آن:

module my-module

go 1.19

require github.com/username/my-lib v1.2.0

replace github.com/username/my-lib => ../local-path/my-lib

در این مثال:

require بیانگر وابستگی به نسخه v1.2.0 کتابخانه my-lib است.
replace به کامپایلر Go می‌گوید به جای استفاده از نسخه ‘v1.2.0‘ در مخزن اصلی، کتابخانه را از مسیر محلی ../local-path/my-lib برداشت کند.

این روش برای توسعه‌دهندگانی که می‌خواهند در عین حال که پروژه‌شان از نسخه رسمی استفاده می‌کند، برخی ویژگی‌ها یا اصلاحات را در یک نسخه محلی آزمایش کنند، بسیار کارآمد است.

استفاده از go mod vendor

در برخی پروژه‌ها (به‌خصوص در سازمان‌های بزرگ یا محیط‌هایی که اتصال مستقیم به اینترنت ندارند)، نیاز است همه وابستگی‌ها به شکل آفلاین در فولدری نگهداری شوند تا ساخت
‘build‘ در هر زمانی و بدون نیاز به دانلود مجدد کتابخانه‌ها قابل انجام باشد. دستور زیر تمام وابستگی‌ها را در پوشه vendor/ کپی می‌کند:

go mod vendor

پوشه vendor/ حاوی تمامی کدهای کتابخانه‌های مورد نیاز پروژه است.
با افزودن گزینه -mod=vendor به فرمان‌های build یا test، می‌توان اطمینان حاصل کرد که تنها از کدهای موجود در vendor/ استفاده می‌شود.
روش vendoring مخصوصاً برای سازمان‌هایی که مایل نیستند به صورت مداوم به اینترنت متصل باشند یا نسخه‌های غیرقابل اطمینان را دانلود کنند، بسیار رایج است.

نکاتی برای کار تیمی با go mod

نسخه‌گذاری منظم: اگر ماژول جداگانه‌ای توسعه می‌دهید که دیگران هم از آن استفاده می‌کنند، نسخه‌گذاری
‘tag‘ منظم در مخزن گیت اهمیت زیادی دارد. با این کار، پروژه‌های مختلف دقیقاً می‌دانند با کدام نسخه از کتابخانه شما سازگار هستند.
همگام‌سازی تغییرات: در تیم‌هایی که چند نفر روی یک پروژه کار می‌کنند، هر بار پس از افزودن کتابخانه جدید یا تغییر نسخه‌ها، اطمینان حاصل کنید که فایل‌های go.mod و go.sum در مخزن گیت به‌روزرسانی شده و دیگر اعضا آن را دریافت کرده‌اند.

بررسی تست‌ها قبل از به‌روزرسانی: هنگام اجرای دستور go get -u …، همیشه بهتر است تمامی تست‌های پروژه ‘gotest−v‘یا‘gotest−cover‘ را اجرا کنید تا مطمئن شوید نسخه جدید کتابخانه‌ها به کد شما لطمه نمی‌زند.
مراقب تغییرات major باشید: در زبان Go، نسخه‌های major (مثلاً از v1 به v2) می‌توانند حاوی تغییرات ناسازگار باشند. قبل از مهاجرت به نسخه‌های
‘major‘، مستندات و لیست تغییرات  ‘changelog‘ آن کتابخانه را مطالعه کنید.
ابزار go mod ستون فقرات مدیریت وابستگی در زبان Go است و ویژگی‌هایی همچون کنترل نسخه دقیق، سادگی در افزودن/حذف کتابخانه‌ها، و قابلیت توسعه لوکال با دستور replace را فراهم می‌کند. تسلط بر این ابزار به شما کمک می‌کند پروژه‌های حرفه‌ای‌تری بسازید و در تیم‌های توسعه، همکاری کارآمدتری داشته باشید. با اتکا به امکانات ماژول‌ها و ابزارهای Go، از سردرگمی‌های مرسوم در مدیریت وابستگی رهایی می‌یابید و تمرکزتان را بر تولید کد باکیفیت و قابل نگهداری قرار خواهید داد.

انتشار ماژول‌ها

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

۱. ایجاد مخزن ‘repository‘ جدید

در نخستین گام، باید پروژه یا ماژول خود را در یک سرویس میزبانی کد نظیر GitHub، GitLab یا Bitbucket قرار دهید. به عنوان مثال، در GitHub می‌توانید یک ریپازیتوری عمومی ‘publicrepository‘ جدید با نام زیر ایجاد کنید:

github.com/username/my-module

نام ریپازیتوری باید توصیف‌کننده‌ی هدف ماژول باشد تا کاربران به‌سرعت ماهیت آن را دریابند.
می‌توانید در توضیحات ‘description‘ پروژه، کارکرد و ویژگی‌های اصلی ماژول را بنویسید.

۲. اضافه کردن کدهای ماژول

در این مرحله، ساختار پروژه Go خود را دقیقاً در پوشه مخزن قرار دهید. توصیه می‌شود حتماً فایل‌های کلیدی ماژول مانند go.mod و go.sum را در سطح اصلی ‘root‘ پروژه نگه دارید تا Go بتواند به‌درستی ماژول را شناسایی کند. برای مثال:

my-module/
  ├── go.mod
  ├── go.sum
  ├── main.go
  ├── utils/
  │   └── helper.go
  └── README.md

README.md: بهتر است توضیحات مختصری از پروژه، نحوه نصب و نمونه کد‘codeexample‘ در این فایل قرار دهید تا توسعه‌دهندگان سریع‌تر با ماژول شما آشنا شوند.
main.go (در صورت وجود): اگر ماژول شما شامل برنامه اجرایی‘executable‘ است،
فایل main.go را در پکیج main تعریف کنید؛ در غیر این صورت، ماژول می‌تواند صرفاً یک کتابخانه باشد.

۳. اصلاح فایل go.mod

اکنون نوبت آن رسیده تا در فایل go.mod، نام ماژول‘modulepath‘ را مطابق آدرس مخزن خود اصلاح کنید.
این کار کمک می‌کند تا دیگران بتوانند به‌صورت مستقیم با دستور go get ماژول شما را دریافت کنند. نمونه فایل go.mod برای یک ماژول در GitHub ممکن است به‌صورت زیر باشد:

module github.com/username/my-module

go 1.19

نام ماژول باید دقیقاً مطابق آدرس مخزن آنلاین شما باشد.
نسخه این زبان  ‘go1.19‘ را می‌توانید با توجه به ورژن‌های پشتیبانی‌شده در پروژه‌تان تغییر دهید.
پس از ویرایش این فایل، اگر تغییراتی در وابستگی‌های پروژه انجام داده‌اید، دستور زیر را برای مرتب‌سازی اجرا کنید:

go mod tidy

این دستور فایل‌های go.mod و go.sum را به‌روزرسانی کرده و وابستگی‌های اضافه را حذف می‌کند.

۴. تگ‌گذاری‘tag‘ و نسخه‌بندی

برای اینکه دیگران بتوانند از نسخه‌های مختلف ماژول شما استفاده کنند، لازم است از تگ‌گذاری‘gittagging‘ و نسخه‌بندی‘versioning‘ بهره ببرید. معمولاً در دنیای Go، از اصول نسخه‌بندی معنایی‘SemVer‘ استفاده می‌شود. مراحل کلی به این شکل است:
ایجاد تگ: به کمک دستورات git در خط فرمان، تگ مورد نظرتان را ایجاد کنید:

git tag v1.0.0

با این فرمان، یک تگ با نسخه v1.0.0 ایجاد می‌شود.

انتشار تگ: در ادامه، تگ به همراه تغییرات کد در مخزن شما آپلود می‌شود:

git push origin v1.0.0

نسخه‌بندی: برای ارائه نسخه‌های بعدی، می‌توانید از الگوی v1.1.0, v1.2.0 و … یا حتی v2.0.0 در صورت تغییرات ناسازگار‘breakingchanges‘ استفاده کنید.
نکته: در صورتی که تغییرات ماژول شما با نسخه پیشین ناسازگار است، لازم است نسخه اصلی‘major‘ را افزایش دهید (برای مثال، از v1.x.x به v2.x.x بروید). این کار کمک می‌کند تا کاربران قبلی ماژول شما بدون اطلاع، دچار مشکلات ناشی از تغییرات ناسازگار نشوند.

۵. انتشار یا به‌روزرسانی

پس از اتمام مراحل فوق و push تغییرات (commit و push فایل‌های پروژه و تگ‌ها**)، ماژول شما به‌صورت عمومی قابل دسترس خواهد بود. توسعه‌دهندگان دیگر می‌توانند به‌راحتی با دستور زیر، ماژول شما را دریافت کنند:

go get github.com/username/my-module

Go به صورت خودکار فایل‌های go.mod و go.sum پروژه مقصد را با اطلاعات ماژول شما به‌روزرسانی می‌کند.
در صورت تغییر یا انتشار نسخه جدید‘tag‘جدید، کافی است مجدداً تغییرات را به مخزن ارسال کنید تا کاربران بتوانند با به‌روزرسانی ماژول
‘goget−u…‘ از نسخه جدید بهره ببرند.

نکات مهم در انتشار ماژول‌ها

مستندات: ارائه مستندات‘documentation‘ و مثال‌های کاربردی در فایل README.md یا در صفحات ویکی‘wiki‘ می‌تواند کار با ماژول شما را برای دیگران آسان کند.
تست‌ها: اضافه کردن تست‌های واحد‘unittests‘ و پوشش‌دهی ‘coverage‘ مناسب، نشان‌دهنده‌ی کیفیت و قابل اعتماد بودن ماژول است.
مدیریت تغییرات: اگر تغییری انجام دادید که با نسخه‌های قبلی ناسازگار است، بهتر است آن را در یک نسخه اصلی‘major‘ جدید منتشر کنید و در توضیحات تگ نسخه، لیست تغییرات‘changelog‘ را برای آگاهی کاربران بیاورید.
استفاده از CI/CD: برای تسهیل فرآیند تست و انتشار، می‌توانید از سیستم‌های یکپارچگی مداوم‘CI‘ مانند GitHub Actions یا GitLab CI استفاده کنید. این سرویس‌ها به‌صورت خودکار کد شما را پس از هر کامیت کامپایل و تست کرده و در صورت موفقیت، می‌توانند تگ‌گذاری و انتشار را نیز آسان‌تر کنند.
با طی کردن مراحل ایجاد مخزن، اضافه کردن کد، اصلاح فایل go.mod، تگ‌گذاری و در نهایت انتشار یا به‌روزرسانی، ماژول شما در اکوسیستم ماژول‌ها و ابزارهای Go قابل دسترس خواهد بود. این قابلیت نه تنها امکان اشتراک‌گذاری کد را فراهم می‌کند، بلکه می‌تواند به ایجاد یک جامعه فعال اطراف ماژول شما منجر شود. کاربران دیگر می‌توانند بازخورد دهند، مشکلات را گزارش کنند یا حتی مشارکت‘contribution‘ داشته باشند و در ادامه باعث بهبود و رشد پروژه شوند. با رعایت این نکات و تکنیک‌ها، ماژول‌هایی قابل اعتماد و پایدار منتشر خواهید کرد که به‌راحتی توسط جامعه Go پذیرش و استفاده می‌شوند.

ابزارهای تست

بخشی جدایی‌ناپذیر از چرخه توسعه، اطمینان از صحت و پایداری کد است. زبان Go با ارائه ابزار تست داخلی، این امکان را فراهم کرده است که بدون نیاز به ابزارهای خارجی، کیفیت کد خود را بررسی کنید. در ادامه، جنبه‌های مختلف استفاده از ابزارهای تست را که بخشی از ماژول‌ها و ابزارهای Go هستند، با جزئیات بیشتری توضیح می‌دهیم.

۱. ساخت فایل تست

اولین گام در فرایند تست، ایجاد یک فایل تست است. اگر قصد دارید محتوای فایل main.go یا پکیج utils/ را تست کنید، می‌توانید با افزودن پسوند _test.go فایل تست بسازید. نمونه‌هایی از آن به‌صورت زیر است:

main_test.go (برای تست فایل main.go در پکیج main)
helper_test.go (برای تست فایل helper.go در پکیج utils)
در این فایل‌ها از پکیج‘package‘ همان پوشه یا پکیج مورد نظرتان استفاده می‌کنید. معمولاً نام فایل تست و فایل اصلی یکسان است و تنها پسوند _test.go اضافه می‌شود تا Go این فایل را به‌عنوان تست شناسایی کند.

۲. نوشتن توابع تست

در زبان Go، تمام توابع تست باید با واژه Test آغاز شوند. این الگو به کامپایلر و ابزار تست Go کمک می‌کند تا به صورت خودکار تمام توابع تست را شناسایی کند. ساختار کلی یک تابع تست در Go به شکل زیر است:

func TestSomething(t *testing.T) {
    // ...
}

در این تابع:

متغیر t از نوع *testing.T است که روش‌ها و متدهایی برای گزارش موفقیت یا شکست تست در اختیار شما قرار می‌دهد.
از متدهای t.Error، t.Errorf، یا t.Fatal برای نشان دادن شرایط خطا استفاده می‌شود.
از سایر توابع (مانند t.Log) می‌توان برای لاگ کردن اطلاعات اضافی در زمان اجرای تست کمک گرفت.
نمونه کد تست:

package main

import (
    "testing"
)

func Hello() string {
    return "Hello Go!"
}

func TestHello(t *testing.T) {
    got := Hello()
    want := "Hello Go!"
    if got != want {
        t.Errorf("Expected %v, got %v", want, got)
    }
}

در این مثال:

تابع Hello() رفتار مورد انتظار ما را برمی‌گرداند: “Hello Go!”
در تابع تست TestHello، مقادیر got و want را مقایسه می‌کنیم و در صورت ناسازگاری، پیام خطا تولید می‌کنیم.

۳. اجرای تست‌ها

برای اجرای همه توابع تست در یک پکیج، کافی است از دستور زیر استفاده کنید:

go test

این دستور تمامی فایل‌های _test.go در پکیج فعلی را جست‌وجو کرده و توابعی را که با Test شروع می‌شوند، اجرا می‌کند. در صورتی که مایل به مشاهده جزئیات بیشتری مانند نام هر تابع و خروجی آن باشید، می‌توانید از پرچم‘flag‘ -v استفاده کنید:

go test -v

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

۴. تست پوشش‘coverage‘

معمولاً توسعه‌دهندگان علاقه‌مند هستند بدانند چه مقدار از کد آن‌ها تحت پوشش تست‌های نوشته شده قرار دارد. این کار به کمک پرچم -cover قابل انجام است:

go test -cover

Go درصد پوشش کل را محاسبه کرده و نمایش می‌دهد. هرچه این عدد بالاتر باشد، یعنی تست‌ها بخش بیشتری از کد را در بر می‌گیرند. اما نباید صرفاً به کمیت پوشش اکتفا کرد؛ تست‌ها باید از نظر کیفی نیز معتبر باشند و رفتارهای غیرمنتظره را ارزیابی کنند.

همچنین اگر می‌خواهید گزارش پوشش دقیقی در قالب فایل داشته باشید، می‌توانید از دستور زیر استفاده کنید:

go test -coverprofile=coverage.out

و سپس با دستور زیر در مرورگر، گزارش را مشاهده نمایید:

go tool cover -html=coverage.out

این گزارش به شما نشان می‌دهد هر خط کدام فایل‌ها تست شده و کدام خط‌های کد تست نشده باقی مانده است.

۵. نکات تکمیلی در تست‌نویسی

تست جدول‌محور‘table−driventests‘: برای نوشتن مجموعه‌ای از ورودی‌ها و خروجی‌های متنوع، به‌جای تعریف چند تابع تست جداگانه، می‌توانید از یک جدول‘slice‘ از ساختار داده استفاده کنید و داده‌ها را در یک حلقه‘loop‘ بررسی کنید. این رویکرد به کدنویسی تمیزتر و مدیریت بهتر سناریوهای متنوع کمک می‌کند.

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

go test ./utils

در این دستور فقط تست‌های موجود در پوشه utils اجرا می‌شوند.

استفاده از Subtests: با استفاده از تابع t.Run(“Name”, func(t *testing.T) { … }) می‌توانید در یک تابع تست، چندین سناریو را با نام‌های متفاوت اجرا کنید. این کار به مشاهده مجزای نتیجه هر سناریو و گزارش بهتر خطاها منجر می‌شود.

تست‌های Benchmark: اگر قصد ارزیابی عملکرد و سرعت اجرای کد را دارید، می‌توانید از توابع بنچمارک‘BenchmarkXXXXX‘ در فایل‌های تست استفاده کنید. اجرای بنچمارک با دستور زیر صورت می‌گیرد:

go test -bench=.

این فرمان تمام توابعی را که با Benchmark شروع می‌شوند، اجرا کرده و نتیجه‘ops/sec‘،‘ns/op‘،وغیره را نمایش می‌دهد.
تست‌های Example: گو در مستندات رسمی و ماژول‌ها و ابزارهای Go قابلیتی به نام تست‌های مثالی‘Example‘ دارد که باعث می‌شود همزمان با تست، مستنداتی قابل استفاده و قابل اجرا هم تولید شوند. ساختار این تست‌ها با Example شروع شده و خروجی استاندارد را با Output: مقایسه می‌کند.

ابزارهای فرمت‌بندی کد

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

۱. ابزار فرمت‌بندی کد (gofmt یا go fmt)

یکی از اصلی‌ترین و پراستفاده‌ترین ابزارهای Go برای فرمت‌بندی کد، دستور gofmt است که در قالب یک ابزار اصلی به نام go fmt نیز ارائه می‌شود. با اجرای این دستور در ترمینال، تمام فایل‌های *.go در پوشه جاری (و زیرفولدرها در صورت استفاده از اسکریپت‌ها یا ابزارهای اضافه) مطابق سبک نگارشی زبان Go قالب‌بندی می‌شوند.

go fmt

صرفه‌جویی در زمان: با تکیه بر این ابزار، دیگر لازم نیست وقت خود را صرف همسان‌سازی فاصله‌ها، تورفتگی‌ها و نظم کد کنید. این کار به‌صورت خودکار انجام می‌شود.
استانداردسازی: gofmt از قواعد رسمی و توصیه‌شده زبان Go استفاده می‌کند. در نتیجه کدی که توسط افراد مختلف نوشته می‌شود، در ظاهر و شیوه نگارش یکپارچه خواهد بود.
افزایش خوانایی: استفاده از فرمت ثابت و یکدست، درک کد را برای تمامی اعضای تیم توسعه و حتی برای شخص نویسنده در آینده، آسان‌تر می‌کند.
نکته: بسیاری از ویرایشگرهای متن‘IDE‘ نظیر Visual Studio Code یا GoLand، این قابلیت را دارند که پس از ذخیره فایل‘onsave‘، به صورت خودکار go fmt را اجرا کرده و کد را مرتب نمایند.

۲. بررسی سبک کدنویسی (golint)

علاوه بر فرمت‌بندی، بررسی سبک کدنویسی‘linting‘ نیز سهم مهمی در کیفیت کلی کد ایفا می‌کند. ابزار golint با تحلیل استایل کد شما، هشدارهایی در خصوص موارد زیر ارائه می‌دهد:
نام‌گذاری ‘namingconventions‘: مثلاً توصیه می‌کند نام توابع، متغیرها و ساختارها‘struct‘ به شکل CamelCase باشد.
نظرات‘comments‘: وجود توضیحات کافی برای توابع عمومی‘exportedfunctions‘ یا بسته‌ها‘packages‘ را بررسی می‌کند.

الگوهای بد احتمالی: هشدار در مورد الگوهای کدنویسی‌ای که ممکن است در آینده به مشکلات منجر شوند.
مثال استفاده از golint:

go install golang.org/x/lint/golint@latest
golint mypackage

خروجی golint: پیام‌هایی حاوی راهنمایی‌هایی است که در بهبود کیفیت و خوانایی کد مفید واقع می‌شوند. هرچند که بعضی از این هشدارها ممکن است سلیقه‌ای به‌نظر برسند، اما در اغلب موارد رعایت این پیشنهادات به بهبود قابل توجهی در نگهداری پروژه منجر می‌شود.

۳. مدیریت خودکار ایمپورت‌ها (goimports)

یکی دیگر از ابزارهای مفیدی که در کنار فرمت‌بندی کد به‌شدت توصیه می‌شود، goimports است. وظیفه اصلی این ابزار، مدیریت و مرتب‌سازی
‘sorting‘ بخش import کدها به صورت خودکار است. وقتی کتابخانه‌ای را از کد حذف می‌کنید، goimports سطر مربوطه در قسمت import را پاک می‌کند. همچنین اگر از کتابخانه جدیدی استفاده کنید، goimports به‌طور خودکار آن کتابخانه را به بخش import اضافه می‌نماید.

نمونه استفاده:

go install golang.org/x/tools/cmd/goimports@latest
goimports -w main.go

با گزینه -w، تغییرات به‌طور مستقیم در فایل main.go اعمال می‌شوند.
goimports علاوه بر مدیریت ایمپورت‌ها، همانند gofmt فایل را فرمت می‌کند و از قوانین نگارشی مشابه پیروی می‌کند.

۴. چرا فرمت‌بندی اتوماتیک و ابزارهای linting مهم‌اند؟

صرفه‌جویی در هزینه‌های نگهداری: هرچه کد تمیزتر و استانداردتر باشد، عیب‌یابی و افزودن ویژگی‌های جدید سریع‌تر و با ریسک کمتر صورت می‌گیرد.
بالا رفتن کیفیت پروژه: استفاده از مجموعه ابزارهایی نظیر gofmt, golint و goimports منجر به کدی می‌شود که در آینده نیز به‌راحتی قابل فهم و تغییر است.
افزایش بهره‌وری تیم: در تیم‌های توسعه، هماهنگ کردن سبک کدنویسی کار دشواری است. این ابزارها با اتوماتیک‌کردن فرایند، تضاد سلایق شخصی را به حداقل می‌رسانند و یکپارچگی را حفظ می‌کنند.
کاهش خطاهای پنهان: برخی مشکلات نگارشی کوچک، ممکن است در آینده دردسرساز شوند. ابزارهای lint می‌توانند آن‌ها را شناسایی کرده و پیش از تبدیل به مشکل حاد، شما را مطلع کنند.

۵. تجمیع ابزارها در فرایند CI/CD

ابزارهایی مانند go fmt, goimports, golint علاوه بر اینکه می‌توانند به‌صورت دستی در سیستم هر توسعه‌دهنده اجرا شوند،‌ در فرایند یکپارچگی مداوم‘ContinuousIntegration‘یاCI نیز بسیار کاربردی هستند. بسیاری از سیستم‌های CI نظیر GitHub Actions یا GitLab CI/CD امکان اجرای این دستورها را در هر commit یا pull request فراهم می‌کنند. نتیجه این کار:

اطمینان از عدم ورود کد غیرفرمت‌شده به مخزن‘repository‘.
خودکار بودن ارزیابی سبک کدنویسی و کیفیت کد قبل از ادغام‘merge‘.
بهبود چرخه توسعه: خطاهایی که به صورت دستی مغفول می‌مانند، به‌سرعت توسط سیستم شناسایی شده و به توسعه‌دهنده گزارش می‌شوند.

ابزارهای فرمت‌بندی و لینتینگ، بخش مهمی از ماژول‌ها و ابزارهای Go را تشکیل می‌دهند و عامل مهمی در حفظ کیفیت و خوانایی کد هستند. در یک جمع‌بندی کلی:

go fmt یا gofmt برای فرمت‌بندی کد؛
golint برای بررسی الگوها و سبک نگارشی مناسب؛
goimports برای مدیریت خودکار ایمپورت‌ها؛
همگی در کنار یکدیگر تجربه توسعه‌ای روان‌تر، باکیفیت‌تر و منظم‌تر را برای شما فراهم می‌کنند. با استفاده از این ابزارها، تمرکز خود را روی منطق کسب‌وکار و پیاده‌سازی ایده‌های اصلی پروژه بگذارید و از صرف وقت برای اصلاحات نگارشی و سبک کدنویسی بی‌نیاز شوید.

کامپایلر Go

یکی دیگر از اجزای کلیدی در ماژول‌ها و ابزارهای Go، کامپایلر Go است. کامپایلر Go وظیفه دارد کد نوشته‌شده در زبان Go را تبدیل به یک باینری قابل اجرا‘executable‘ کند و فرایند ساخت ‘build‘ و اجرای پروژه را تسهیل نماید. ازآنجاکه Go زبان کامپایل شونده‘compiledlanguage‘ است، خروجی حاصل از کامپایل می‌تواند در سیستم‌عامل‌های مختلف ‘cross−compile‘ اجرا شود. در ادامه، مهم‌ترین دستورات و قابلیت‌های مرتبط با کامپایلر Go را مرور می‌کنیم.

۱. ساخت و اجرای سریع کد (go run)

اگر بخواهید به‌سرعت یک فایل‘main.go‘ را اجرا کنید، بدون آنکه به‌صورت جداگانه فایل باینری ساخته شود، دستور زیر راهگشاست:

go run main.go

کامپایل سریع: این دستور، سورس‌کد را در لحظه کامپایل کرده و بدون نگهداری فایل باینری، آن را اجرا می‌کند.
مناسب برای توسعه اولیه: در مراحل اولیه توسعه و آزمایش سریع کد، اغلب از go run استفاده می‌شود تا نیازی به ساخت و نگهداری باینری نباشد.
پشتیبانی از چند فایل: اگر برنامه شما در چندین فایل *.go پخش شده باشد، می‌توانید همه آن‌ها را در یک دستور فراخوانی کنید. برای نمونه:

go run main.go utils.go

۲. ساخت باینری (go build)

زمانی که پروژه شما به حدی رسیده باشد که نیاز به یک خروجی مستقل‘executable‘ داشته باشید یا قصد توزیع آن را دارید، از دستور go build استفاده کنید:

go build

ساخت باینری: دستور go build تمام فایل‌های پروژه (به‌ویژه آن‌هایی که در پکیج main قرار دارند) را کامپایل و یک فایل اجرایی‘executablefile‘ در سیستم‌عامل میزبان تولید می‌کند.

نام خروجی: در صورت وجود فایل main.go، نام فایل خروجی معمولاً با نام پوشه‘directory‘ پروژه یکی است. می‌توانید با استفاده از فلگ‘flag‘ -o نام و مسیر خروجی را مشخص کنید. مثلاً:

go build -o myapp

این دستور یک فایل اجرایی با نام myapp می‌سازد.
بدون اجرای کد: برخلاف go run که پس از کامپایل پروژه را اجرا می‌کند، go build تنها خروجی را می‌سازد و اجرا نمی‌کند.

۳. ساخت برای سیستم‌عامل‌ها و معماری‌های مختلف (Cross-Compile)

یکی از قابلیت‌های برجسته زبان Go، توانایی تولید خروجی قابل اجرا برای سیستم‌عامل‌ها و معماری‌های متفاوت است. با تعیین متغیرهای محیطی
‘environmentvariables‘ مناسب، می‌توانید از یک سیستم‘host‘ برای ساخت باینری هدف ‘target‘ استفاده کنید. مثال زیر ساخت یک باینری برای لینوکس ۶۴بیتی را نشان می‌دهد، حتی اگر سیستم فعلی شما ویندوز یا مک باشد:

GOOS=linux GOARCH=amd64 go build

GOOS: تعیین‌کننده سیستم‌عامل هدف ‘linux‘,‘windows‘,‘darwin‘برایمک،وغیره.
GOARCH: تعیین‌کننده معماری پردازنده‘amd64‘,‘386‘,‘arm‘,‘arm64‘,وغیره.

کاربرد: برای توزیع برنامه‌های Go در محیط‌های مختلف (مثلاً سرورهای لینوکسی، دستگاه‌های ARM یا سیستم‌های ویندوزی)، دیگر نیازی به ساخت پروژه روی آن سیستم‌ها به‌صورت جداگانه نخواهید داشت. تنها کافی است با مقداردهی این متغیرهای محیطی، یک بار روی سیستم خودتان باینری مربوطه را بسازید.

۴. نکات و ابزارهای تکمیلی در ساخت و اجرا

استفاده از فلگ‌های مختلف: Go اجازه می‌دهد در هنگام کامپایل گزینه‌های مختلفی را تعیین کنید؛ برای مثال:
-race: برای شناسایی شرایط رقابتی
‘raceconditions‘ در برنامه‌های همزمان.
-ldflags: برای تنظیم لینکینگ
‘linking‘، مانند درج اطلاعات نسخه در برنامه.
-gcflags: برای تعیین فلگ‌های مربوط به کامپایلر و اشکال‌زدایی عمیق‌تر.
جداسازی پکیج‌ها: وقتی پروژه شما بزرگ است، ممکن است کدها در چندین پکیج و پوشه ذخیره شوند. در این حالت، go build به شکل پیش‌فرض پکیج‌های فرزند‘childpackages‘ را هم کامپایل می‌کند.
‘cache‘: از نسخه‌های جدید Go به بعد، یک مکانیزم کش‘buildcache‘ وجود دارد که باعث می‌شود اگر کدی تغییر نکرده، دوباره از ابتدا کامپایل نشود و سرعت فرایند ساخت افزایش یابد.

مختصری درباره go install: این دستور نیز مشابه go build عمل می‌کند اما خروجی را در محل تعیین‌شده  یامحل تنظیم‌شدهدرمتغیرمحیطی
‘GOPATH/bin‘یامحلتنظیم‌شدهدرمتغیرمحیطی قرار می‌دهد. اغلب برای نصب ابزارهای مختلف Go به کار می‌رود.

کامپایلر Go هسته اصلی فرایند ساخت و اجرای کد در پروژه‌های Go است.
دستورات متنوعی نظیر go run, go build و قابلیت Cross-Compile به توسعه‌دهندگان اجازه می‌دهد تا به سادگی و سرعت، پروژه‌هایشان را روی سیستم محلی اجرا کرده یا باینری قابل‌حمل برای پلتفرم‌های مختلف ایجاد کنند.
این ویژگی‌ها، در کنار سایر قابلیت‌های ماژول‌ها و ابزارهای Go مثل مدیریت ماژول‌ها و وابستگی‌ها، تست و فرمت‌بندی کد، چرخه توسعه کاملی را برای رسیدن به محصول نهایی فراهم می‌کنند.
با تسلط بر این دستورات و امکانات، می‌توانید پروژه‌های Go خود را حرفه‌ای‌تر و کارآمدتر توسعه داده و در طیف گسترده‌ای از محیط‌ها اجرا نمایید.

نتیجه‌گیری

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

آموزش ماژول‌ها و ابزارهای Go

دیدگاه های شما

دیدگاهتان را بنویسید

نشانی ایمیل شما منتشر نخواهد شد. بخش‌های موردنیاز علامت‌گذاری شده‌اند *