شما در حال مشاهده اسناد Apigee Edge هستید.
به مستندات Apigee X بروید . اطلاعات
دستورات شرطی یک ساختار کنترلی مشترک در همه زبان های برنامه نویسی هستند. مانند یک زبان برنامه نویسی، پیکربندی پراکسی API از عبارات شرطی برای جریان ها، سیاست ها، مراحل و روت رول ها پشتیبانی می کند. با تعریف عبارات شرطی، رفتار پویا را برای API خود تعریف می کنید. این رفتار پویا به شما امکان میدهد کارهایی مانند تبدیل XML به JSON فقط برای دستگاههای تلفن همراه یا مسیریابی به URL پشتیبان بر اساس نوع محتوا یا فعل HTTP پیام درخواست را انجام دهید.
این مبحث به شما نشان میدهد که چگونه میتوانید از شرایط برای اعمال پویا ویژگیهای مدیریت API در زمان اجرا، بدون نوشتن هیچ کدی استفاده کنید.
دستورات شرطی را پیکربندی کنید
رفتار شرطی در پراکسی های API با استفاده از ترکیبی از شرایط و متغیرها پیاده سازی می شود. یک دستور شرطی با استفاده از عنصر Condition ایجاد می شود. شرط زیر یک شرط خالی است:
<Condition></Condition>
برای ایجاد یک دستور شرطی، یک عملگر شرطی و یک متغیر برای ایجاد ساختار زیر اضافه می کنید:
<Condition>{variable.name}{operator}{"value"}</Condition>
عملگرهای شرطی پشتیبانی شده عبارتند از =
(برابر)، !=
(برابر نیست)، و >
(بیشتر از). برای خوانایی، میتوانید شرطها را نیز به صورت متن بنویسید: equals
، notequals
، greaterthan
.
هنگام کار با مسیرهای URI می توانید از ~/
یا MatchesPath
استفاده کنید. همچنین می توانید عبارات منظم JavaRegex را با عملگر ~~ مطابقت دهید.
شرایط برای تعریف جریانهای شرطی پروکسی API برای منابع API پشتیبان، که در ایجاد جریانهای شرطی به منابع API باطنی توضیح داده شده است، استفاده میشود. برای فهرست کامل شرط ها، به مرجع شرایط مراجعه کنید.
متغیرها
شرایط کار خود را با ارزیابی مقادیر متغیرها انجام می دهند. یک متغیر ویژگی یک تراکنش HTTP است که توسط یک پراکسی API اجرا می شود، یا یک ویژگی از خود پیکربندی پروکسی API است. هر زمان که یک پروکسی API درخواستی از یک برنامه دریافت می کند، Apigee Edge لیست طولانی از متغیرهایی را پر می کند که با مواردی مانند زمان سیستم، اطلاعات شبکه برنامه، هدرهای HTTP روی پیام ها، پیکربندی پروکسی API، اجرای سیاست ها و غیره مرتبط هستند. این یک زمینه غنی ایجاد می کند که می توانید از آن برای تنظیم عبارات شرطی استفاده کنید.
متغیرها همیشه از علامت نقطهگذاری استفاده میکنند. برای مثال، هدرهای HTTP در پیام درخواست به عنوان متغیرهایی به نام request.header.{header_name}
. بنابراین برای ارزیابی هدر Content-type، می توانید از متغیر request.header.Content-type
استفاده کنید. برای مثال request.header.Content-type = "application/json"
نشان می دهد که نوع محتوای درخواست باید JSON باشد.
تصور کنید که باید یک عبارت شرطی ایجاد کنید که باعث شود یک خط مشی فقط زمانی اجرا شود که یک پیام درخواست GET باشد. برای ایجاد شرطی که فعل HTTP یک درخواست را ارزیابی می کند، عبارت شرطی زیر را ایجاد می کنید. متغیر در این شرط request.verb
است. مقدار متغیر GET
است. عملگر =
است.
<Condition>request.verb = "GET"</Condition>شما همچنین می توانید استفاده کنید:
<Condition>request.verb equals "GET"</Condition>
Edge از چنین عبارتی برای ارزیابی شرایط استفاده می کند. اگر فعل HTTP مرتبط با درخواست GET باشد، مثال بالا درست ارزیابی میکند. اگر فعل HTTP مرتبط با درخواست POST باشد، عبارت به نادرست ارزیابی می شود.
برای فعال کردن رفتار پویا، میتوانید Conditions را به Flows، Steps و RouteRules متصل کنید.
هنگامی که یک شرط را به یک جریان متصل می کنید، یک "جریان شرطی" ایجاد می کنید. جریان های شرطی فقط زمانی اجرا می شوند که شرط به درستی ارزیابی شود. میتوانید هر تعداد سیاستی که میخواهید به یک جریان شرطی پیوست کنید. یک جریان شرطی به شما امکان میدهد تا قوانین پردازش بسیار تخصصی را برای پیامهای درخواست یا پاسخی که معیارهای خاصی را برآورده میکنند، ایجاد کنید.
به عنوان مثال، برای ایجاد یک Flow که فقط زمانی اجرا می شود که فعل درخواست GET باشد:
<Flows> <Flow name="ExecuteForGETs"> <Condition>request.verb="GET"</Condition> </Flow> </Flows>
برای ایجاد یک جریان برای GET و دیگری برای POST:
<Flows> <Flow name="ExecuteForGETs"> <Condition>request.verb="GET"</Condition> </Flow> <Flow name="ExecuteForPOSTs"> <Condition>request.verb="POST"</Condition> </Flow> </Flows>
همانطور که در مثال زیر نشان داده شده است، می توانید این شرط را در خود Policy Step اعمال کنید. شرط زیر باعث می شود که سیاست VerifyApiKey فقط در صورتی اجرا شود که یک پیام درخواستی POST باشد.
<PreFlow name="PreFlow"> <Request> <Step> <Condition>request.verb equals "POST"</Condition> <Name>VerifyApiKey</Name> </Step> </Request> </PreFlow>
هنگامی که چنین جریانهای شرطی را تعریف کردید، میتوانید خطمشیهایی را به آنها ضمیمه کنید و به یک پراکسی API این امکان را میدهید که مجموعهای از خطمشیها را برای درخواستهای GET و مجموعهای دیگر از خطمشیها را برای درخواستهای POST اعمال کند.
برای اطلاعات جامع مرجع، به منابع زیر مراجعه کنید:
مثال 1
مثال زیر یک جریان شرطی منفرد به نام Convert-for-devices
را نشان می دهد که در جریان پاسخ ProxyEndpoint پیکربندی شده است. شرط را به عنوان عنصری به موجودی که شرط در مورد آن اعمال می شود، اضافه کنید. در این مثال، شرط جزئی از جریان است. بنابراین، هر زمان که عبارت درست ارزیابی شود، جریان اجرا خواهد شد.
<Flows> <Flow name="Convert-for-devices"> <Condition>(request.header.User-Agent = "Mozilla")</Condition> <Response> <Step><Name>ConvertToJSON</Name></Step> </Response> </Flow> </Flows>
برای هر درخواست دریافت شده از یک برنامه، Edge مقادیر تمام هدرهای HTTP موجود به عنوان متغیر را ذخیره می کند. اگر درخواست حاوی یک هدر HTTP به نام User-Agent
باشد، آن هدر و مقدار آن به عنوان متغیری به نام request.header.User-Agent
ذخیره می شود.
با توجه به پیکربندی ProxyEndpoint در بالا، Edge مقدار متغیر request.header.User-Agent
را بررسی می کند تا ببیند آیا شرط به درستی ارزیابی می شود یا خیر.
اگر شرط به درستی ارزیابی شود، یعنی مقدار متغیر request.header.User-Agent
برابر با Mozilla
باشد، آنگاه Flow شرطی اجرا می شود و خط مشی XMLtoJSON به نام ConvertToJSON
اجرا می شود. در غیر این صورت، Flow اجرا نمی شود و پاسخ XML بدون تغییر (در قالب XML) به برنامه درخواست کننده بازگردانده می شود.
مثال 2
بیایید از یک مثال خاص استفاده کنیم که در آن باید پیام پاسخ را از XML به JSON تبدیل کنید - اما فقط برای دستگاه های تلفن همراه. ابتدا خط مشی را ایجاد کنید که پاسخ فرمت XML را از Weather API به JSON تبدیل کند:
<XMLToJSON name="ConvertToJSON"> <Options> </Options> <OutputVariable>response</OutputVariable> <Source>response</Source> </XMLToJSON>
پیکربندی خطمشی بالا به پراکسی API میگوید که پیام پاسخ را دریافت کند، با تنظیمات پیشفرض، تبدیل از XML به JSON را انجام دهد و سپس نتیجه را در پیام پاسخ جدید بنویسد. (اگر پیام درخواستی را از XML به JSON تبدیل می کنید، به سادگی هر دوی این مقادیر را برای request
تنظیم کنید.)
از آنجایی که میخواهید پاسخها را از XML به JSON تبدیل کنید، باید یک جریان پاسخ شرطی را برای انجام تبدیل پیکربندی کنید. به عنوان مثال، برای تبدیل همه پاسخها از XML به JSON قبل از اینکه به برنامه مشتری بازگردانده شوند، جریان پاسخ ProxyEndpoint زیر را پیکربندی کنید.
<Flows> <Flow name="Convert-for-devices"> <Response> <Step><Name>ConvertToJSON</Name></Step> </Response> </Flow> </Flows>
وقتی API را با استفاده از درخواست استاندارد فراخوانی میکنید، پاسخ در JSON قالببندی میشود.
با این حال، هدف شما این است که گزارشهای آب و هوا را تنها زمانی به JSON تبدیل کنید که مشتری درخواستکننده یک دستگاه تلفن همراه باشد . برای فعال کردن چنین رفتار پویا، باید یک دستور شرطی به Flow اضافه کنید.
جریان مشروط را آزمایش کنید
در این درخواست نمونه، هدر HTTP User-Agent
روی Mozilla
تنظیم شده است که باعث می شود عبارت شرطی به درستی ارزیابی شود و جریان شرطی Convert-for-devices
اجرا شود.
$ curl -H "User-Agent:Mozilla" http://{org_name}-test.apigee.net/weather/forecastrss?w=12797282
یا برای چاپ زیبا در جایی که پایتون در دسترس است:
$ curl -H "User-Agent:Mozilla" http://{org_name}-test.apigee.net/weather/forecastrss?w=12797282 | python -mjson.tool
نمونه پاسخ:
. . . "yweather_forecast": [ { "code": "11", "date": "12 Dec 2012", "day": "Wed", "high": "55", "low": "36", "text": "Showers" }, { "code": "32", "date": "13 Dec 2012", "day": "Thu", "high": "56", "low": "38", "text": "Sunny" } ] } . . .
درخواستی که بدون سربرگ User-Agent
یا با مقداری متفاوت از Mozilla
ارسال شود، منجر به پاسخی با قالب XML خواهد شد.
$ curl http://{org_name}-test.apigee.net/weather/forecastrss?w=12797282
پاسخ XML اصلاح نشده برگردانده می شود.
نمونه پاسخ:
<yweather:forecast day="Wed" date="12 Dec 2012" low="36" high="55" text="Showers" code="11" /> <yweather:forecast day="Thu" date="13 Dec 2012" low="38" high="56" text="Sunny" code="32" />
تطبیق الگو
این بخش نحوه استفاده از تطبیق الگو با شرایط در یک جریان Apigee را توضیح می دهد.
اپراتورها
این بخش نحوه استفاده از عملگرهای تطبیق الگوی زیر را در عبارات شرطی توضیح می دهد:
- عملگر مسابقات : تطبیق الگوی ساده
- عملگر JavaRegex : کنترل دقیق تر بر تطبیق
- عملگر MatchesPath : تطبیق قطعه مسیر
مسابقات
بیایید ابتدا به عملگر شرطی "Matches" یا "~" نگاه کنیم. این دو عملگر یکسان هستند -- نسخه انگلیسی "Matches" گزینه قابل خواندن تری در نظر گرفته می شود.
خلاصه: عملگر "Match" دو احتمال را به شما می دهد. یا رشته را به معنای واقعی کلمه مطابقت دهید، یا یک تطبیق عام با "*" انجام دهید. همانطور که ممکن است انتظار داشته باشید، حروف عام با صفر یا بیشتر کاراکتر مطابقت دارد. بیایید ببینیم این چگونه کار می کند.
XML زیر یک شرط Step را نشان می دهد. زمانی سیاست SomePolicy را اجرا می کند که شرط به درستی ارزیابی شود. در این مثال، متغیر proxy.pathsuffix
را آزمایش میکنیم، یک متغیر داخلی در Edge که پسوند مسیر درخواست را ذخیره میکند. توجه داشته باشید، با این حال، می توانید مقدار هر متغیر جریانی را که دارای یک رشته است، آزمایش کنید. بنابراین، در این حالت، اگر مسیر پایه درخواست ورودی /animals
و درخواست /animals/cat
باشد، پسوند مسیر رشته تحت اللفظی " /cat
" است.
<PreFlow name="PreFlow"> <Request> <Step> <Condition>(proxy.pathsuffix Matches "/cat")</Condition> <Name>SomePolicy</Name> </Step> </Request> <Response/> </PreFlow>
سوال: کدام پسوند مسیر پراکسی باعث اجرای SomePolicy می شود؟ فقط یک امکان وجود دارد
تماس API:
GET http://artomatic-test.apigee.net/matchtest/cat
آیا سیاست اجرا می شود؟ بله، زیرا پسوند مسیر پراکسی دقیقاً با " /cat
" مطابقت دارد. اگر پسوند /bat
یا /dog
یا " /
" یا هر چیز دیگری باشد، اجرا نمی شود.
اکنون، این عبارت شرطی را در نظر بگیرید که در آن از کاراکتر عام " *
" استفاده می کنیم:
<Condition>(proxy.pathsuffix Matches "/*at")</Condition>
تماس API:
GET http://artomatic-test.apigee.net/matchtest/cat
آیا سیاست اجرا می شود؟ بله، زیرا حروف عام با هر شخصیتی مطابقت دارد و "/cat
" یک تطابق است.
تماس API:
GET http://artomatic-test.apigee.net/matchtest/bat
آیا سیاست اجرا می شود؟ بله، چون حروف عام با هر شخصیتی مطابقت دارد، "/bat"
یک تطابق است.
تماس API:
GET http://artomatic-test.apigee.net/matchtest/owl
آیا سیاست اجرا می شود؟ مطمئناً نه -- اگرچه حروف عام با " o
" مطابقت دارد، اما حروف " wl
" مطابقت ندارند.
حال اجازه دهید علامت عام را به انتهای پسوند منتقل کنیم:
<Condition>(proxy.pathsuffix Matches "/cat*")</Condition>
تماس API:
GET http://artomatic-test.apigee.net/matchtest/cat
آیا سیاست اجرا می شود؟ بله، زیرا حروف عام با صفر یا بیشتر از هر کاراکتری مطابقت دارد.
تماس API:
GET http://artomatic-test.apigee.net/matchtest/bat
آیا سیاست اجرا می شود؟ نه، " /bat
" همسان نیست.
تماس API:
GET http://artomatic-test.apigee.net/matchtest/cat123
آیا سیاست اجرا می شود؟ بله، حروف عام با صفر یا بیشتر از هر کاراکتر مطابقت دارد، بنابراین " 123
" یک تطابق ایجاد می کند.
تماس API:
GET http://artomatic-test.apigee.net/matchtest/cat/bird/mouse
آیا سیاست اجرا می شود؟ بله، از آنجا که کاراکتر عام با صفر یا بیشتر از هر کاراکتر مطابقت دارد، بنابراین " /bird/mouse
" یک تطابق ایجاد می کند. توجه داشته باشید که چگونه عبارتی مانند این می تواند شما را به دردسر بیاندازد زیرا با همه چیز بعد از کاراکترهای تحت اللفظی مطابقت دارد!
سوال: آیا عملگر Maches به حروف کوچک و بزرگ حساس است؟
بله. فرض کنید شرایطی مانند این دارید:
<Condition>(proxy.pathsuffix Matches "/*At")</Condition>
تماس API:
GET http://artomatic-test.apigee.net/matchtest/cat
آیا سیاست اجرا می شود؟ خیر، حروف عام با هر حرفی مطابقت دارد (بدون توجه به حروف کوچک)، اما حروف کوچک "a" با "A" مطابقت ندارد.
تماس API:
GET http://artomatic-test.apigee.net/matchtest/bAt
آیا سیاست اجرا می شود؟ بله، مورد مطابقت دارد.
سوال: چگونه با عملگر Maches از کاراکترها فرار کنم؟
از نویسه درصد "%" برای فرار از کاراکترهای رزرو شده استفاده کنید. به عنوان مثال:
<Condition>(proxy.pathsuffix Matches "/c%*at")</Condition>
تماس API:
GET http://artomatic-test.apigee.net/matchtest/cat
آیا سیاست اجرا می شود؟ نه، عملگر Matches به دنبال رشته تحت اللفظی "c*at" است.
تماس API:
GET http://artomatic-test.apigee.net/matchtest/c*at
سوال: آیا این خط مشی اجرا می شود؟
بله، این مسیر، اگرچه کمی غیر معمول، مطابقت دارد.
JavaRegex
همانطور که می بینید، عملگر "Match" برای موقعیت های ساده عالی است. اما شما می توانید از یک عملگر دیگر، "JavaRegex" یا "~~" استفاده کنید. این دو عملگر یکسان هستند، با این تفاوت که JavaRegex قابل خواندن تر است. JavaRegex نامیده می شود زیرا امکان تطبیق الگوی عبارت منظم را می دهد و Edge از قوانین مشابه کلاس های بسته java.util.regex در زبان جاوا پیروی می کند. نحوه کار اپراتور JavaRegex با عملگر Matches بسیار متفاوت است، بنابراین مهم است که این دو را اشتباه نگیرید!
خلاصه: عملگر "JavaRegex" به شما امکان می دهد از نحو عبارت منظم در دستورات شرطی استفاده کنید.
کد زیر یک شرط Step را نشان می دهد. اگر شرط به درستی ارزیابی شود، خط مشی SomePolicy را اجرا می کند. در این مثال، متغیر proxy.pathsuffix
را آزمایش میکنیم، یک متغیر داخلی در Edge که پسوند مسیر درخواست را ذخیره میکند. اگر مسیر پایه درخواست ورودی /animals
و درخواست /animals/cat
باشد، پسوند مسیر رشته تحت اللفظی " /cat
" است.
<PreFlow name="PreFlow"> <Request> <Step> <Condition>(proxy.pathsuffix JavaRegex "/cat")</Condition> <Name>SomePolicy</Name> </Step> </Request> <Response/> </PreFlow>
سوال: کدام پسوند مسیر پراکسی باعث اجرای SomePolicy می شود؟ درست مانند عملگر Maches، در این مورد نیز تنها یک امکان وجود دارد.
تماس API:
GET http://artomatic-test.apigee.net/matchtest/cat
آیا سیاست اجرا می شود؟ بله، زیرا پسوند مسیر پراکسی دقیقاً با " /cat
" مطابقت دارد. اگر پسوند /bat
یا /dog
یا هر چیز دیگری باشد اجرا نمی شود.
حال، بیایید با استفاده از کمیت "*" یک عبارت منظم ایجاد کنیم. این کمیت با صفر یا بیشتر از کاراکتر قبلی مطابقت دارد.
<Condition>(proxy.pathsuffix JavaRegex "/c*t")</Condition>
تماس API:
GET http://artomatic-test.apigee.net/matchtest/cat
آیا سیاست اجرا می شود؟ نه! کمیت "*" با صفر یا بیشتر از کاراکتر قبلی مطابقت دارد که یک " c
" است.
تماس API:
GET http://artomatic-test.apigee.net/matchtest/ccccct
آیا سیاست اجرا می شود؟ بله، زیرا حروف عام با صفر یا بیشتر از کاراکتر قبلی مطابقت دارد.
در مرحله بعد، از کمیت " ?
" استفاده می کنیم که یک بار یا اصلاً با کاراکتر قبلی مطابقت دارد.
<Condition>(proxy.pathsuffix JavaRegex "/ca?t")</Condition>
تماس API:
GET http://artomatic-test.apigee.net/matchtest/cat
آیا سیاست اجرا می شود؟ بله. کمیت " ?
" با صفر یا یک نویسه قبلی مطابقت دارد که یک " a
" است.
تماس API:
GET http://artomatic-test.apigee.net/matchtest/ct
آیا سیاست اجرا می شود؟ بله. کمیت " ?
" با یک یا هیچ یک از نویسه های قبلی مطابقت دارد. در این مورد، هیچ کاراکتر "a" وجود ندارد، بنابراین شرط به درستی ارزیابی می شود.
تماس API:
GET http://artomatic-test.apigee.net/matchtest/caat
آیا سیاست اجرا می شود؟ نه. "؟" کمیت با یکی از کاراکترهای قبلی که " a
" است مطابقت دارد.
در مرحله بعد، ما از سبک " [abc]
" یا "گروهبندی" بیان regex استفاده میکنیم. با کاراکترهای " a
" یا " b
" یا " c
" مطابقت دارد.
<Condition>(proxy.pathsuffix JavaRegex "/[cbr]at")</Condition>
تماس API:
GET http://artomatic-test.apigee.net/matchtest/cat
آیا سیاست اجرا می شود؟ بله. ما در اینجا از عبارات منظم استفاده می کنیم و عبارت " [cbr]
" با "c"، "b"، یا "r" مطابقت دارد. این تماس ها نیز مطابقت دارند:
GET http://artomatic-test.apigee.net/matchtest/bat
GET http://artomatic-test.apigee.net/matchtest/rat
اما این یک مسابقه نیست:
GET http://artomatic-test.apigee.net/matchtest/mat
سوال: آیا عملگر JavaRegex به حروف کوچک و بزرگ حساس است؟
بله. فرض کنید شرایطی مانند این دارید:
<Condition>(proxy.pathsuffix JavaRegex "/ca?t")</Condition>
تماس API:
GET http://artomatic-test.apigee.net/matchtest/cat
آیا سیاست اجرا می شود؟ بله، regex با صفر یا یک کاراکتر قبلی که "a" است مطابقت دارد.
تماس API:
GET http://artomatic-test.apigee.net/matchtest/cAt
سوال: آیا سیاست اجرا می شود؟
خیر، زیرا بزرگ "A" با حروف کوچک "a" مطابقت ندارد.
MatchesPath
عملگر MatchesPath
نیز می توان مانند این "~/" مشخص کرد. کمی شبیه عملگرهای Matches
(~) و JavaRegex (~~) است. اما MatchesPath کاملاً متفاوت است.
فقط به یاد داشته باشید که این عملگر به یک مسیر به عنوان یک سری قطعات نگاه می کند. بنابراین، اگر مسیر: /animals/cats/wild
باشد، میتوانید مسیر را متشکل از بخشهای " /animals
"، " /cats
" و " /wild
" در نظر بگیرید.
عملگر MatchesPath
به شما امکان می دهد از دو علامت عام استفاده کنید: یک ستاره (*) و یک ستاره دوتایی (**). ستاره منفرد با یک عنصر مسیر مطابقت دارد. ستاره دوتایی با یک یا چند عنصر مسیر مطابقت دارد.
بیایید به یک مثال نگاه کنیم. در این مثال، متغیر proxy.pathsuffix
را آزمایش میکنیم، یک متغیر داخلی در Edge که پسوند مسیر درخواست را ذخیره میکند. توجه داشته باشید، با این حال، می توانید مقدار هر متغیر جریانی را که دارای یک رشته است، آزمایش کنید.
<PreFlow name="PreFlow"> <Request> <Step> <Condition>(proxy.pathsuffix MatchesPath "/animals/*")</Condition> <Name>SomePolicy</Name> </Step> </Request> <Response/> </PreFlow>
سوال: کدام پسوند مسیر پراکسی باعث اجرای SomePolicy می شود؟
تماس API:
GET http://artomatic-test.apigee.net/matchtest/animals
سوال: آیا سیاست اجرا می شود؟
خیر، زیرا شرط نیاز به عنصر مسیر دیگری بعد از " /animals
" دارد، همانطور که توسط " /*
" مشخص شده است.
تماس API:
GET http://artomatic-test.apigee.net/matchtest/animals
/
آیا سیاست اجرا می شود؟ بله، مسیر دارای عنصر مسیر دیگری است (قسمت بعد از " /animals/
")، اما فقط خالی است.
تماس API:
GET http://artomatic-test.apigee.net/matchtest/animals/cats
آیا سیاست اجرا می شود؟ بله، زیرا مسیر به وضوح دارای یک عنصر (" /cats
") است که بعد از " /animals
" می آید.
تماس API:
GET http://artomatic-test.apigee.net/matchtest/animals/cats/wild
سوال: آیا سیاست اجرا می شود؟
خیر، زیرا ستاره تنها با یک عنصر مسیر مطابقت دارد و این API بیش از یک عنصر بعد از " /animals
" دارد.
حالا بیایید از ستاره دوتایی استفاده کنیم:
<PreFlow name="PreFlow"> <Request> <Step> <Condition>(proxy.pathsuffix MatchesPath "/animals/**")</Condition> <Name>SomePolicy</Name> </Step> </Request> <Response/> </PreFlow>
سوال: کدام پسوند مسیر پراکسی باعث اجرای SomePolicy می شود؟
تماس API:
GET http://artomatic-test.apigee.net/matchtest/animals
آیا سیاست اجرا می شود؟ خیر، زیرا شرط حداقل به یک عنصر مسیر زیر نیاز دارد که توسط " /**
" مشخص شده باشد.
تماس API:
GET http://artomatic-test.apigee.net/matchtest/animals
/
آیا سیاست اجرا می شود؟
بله، مسیر دارای عنصر مسیر دیگری است (قسمت بعد از " /animals/
")، اما فقط خالی است.
تماس API:
GET http://artomatic-test.apigee.net/matchtest/animals/cats
آیا سیاست اجرا می شود؟
بله، زیرا مسیر حداقل یک عنصر دارد که بعد از " /animals
" می آید.
تماس API:
GET http://artomatic-test.apigee.net/matchtest/animals/cats/wild
آیا سیاست اجرا می شود؟
بله، زیرا مسیر بیش از یک عنصر دارد که بعد از " /animals
" می آید.
اختلاط ستاره ها
میتوانید از ترکیبهای ستاره تک (*) و دوگانه (**) برای اصلاح بیشتر تطابق مسیر خود استفاده کنید.
<PreFlow name="PreFlow"> <Request> <Step> <Condition>(proxy.pathsuffix MatchesPath "/animals/*/wild/**")</Condition> <Name>SomePolicy</Name> </Step> </Request> <Response/> </PreFlow>
تماس API:
همه این تماسهای API یک تطابق ایجاد میکنند:
GET http://artomatic-test.apigee.net/matchtest/animals/cats/wild/
و
GET http://artomatic-test.apigee.net/matchtest/animals/dogs/wild/austrailian
و
GET http://artomatic-test.apigee.net/matchtest/animals/birds/wild/american/finches
منابع API
خدمات RESTful مجموعه ای از منابع API هستند. یک منبع API یک قطعه مسیر URI است که نهادی را شناسایی می کند که توسعه دهندگان می توانند با فراخوانی API شما به آن دسترسی داشته باشند. به عنوان مثال، اگر سرویس شما گزارش های آب و هوا و پیش بینی های آب و هوا را ارائه می دهد، سرویس پشتیبان شما ممکن است دو منبع API را تعریف کند:
- http://mygreatweatherforecast.com /reports
- http://mygreatweatherforecast.com /forecasts
هنگامی که یک پراکسی API ایجاد می کنید (همانطور که در ساخت اولین پروکسی API خود نشان داده شده است)، حداقل یک URL پایه نام مستعار ایجاد می کنید که به سرویس باطن شما نگاشت می شود. به عنوان مثال:
URL پایه Backend | URL پروکسی API جدید/معادل |
---|---|
http://mygreatweatherforecast.com | http://{your_org}-{environment}.apigee.net/mygreatweatherforecast |
در این مرحله میتوانید با استفاده از هر یک از URLهای پایه، با پشتیبان خود تماسهای API برقرار کنید. اما وقتی از URL پروکسی API استفاده می کنید، چیزها شروع به جالب شدن می کنند.
علاوه بر تجزیه و تحلیل API که Edge هنگام استفاده از پراکسی API شروع به جمعآوری میکند، پراکسیها همچنین به شما امکان میدهند جریانهای شرطی را تعریف کنید که به منابع موجود در باطن شما نگاشت میشوند. در اصل، "اگر یک تماس GET به منبع /reports وارد شود، Edge باید کاری انجام دهد."
تصویر زیر تفاوت رفتار بین دو URL را نشان می دهد که در نهایت به یک Backend دسترسی دارند. یکی URL منبع بدون پروکسی است، دیگری یک پروکسی Edge API با یک جریان مشروط به همان منبع باطن است. در زیر جریانهای مشروط را با جزئیات بیشتری شرح خواهیم داد.
چگونه پراکسی های API به منابع باطن خاص نگاشت می شوند
با یک URL پروکسی API که به URL اصلی سرویس پشتیبان نگاشت شده است (هنگامی که پروکسی را ایجاد می کنید)، می توانید جریان های مشروط را به منابع خاصی اضافه کنید، مانند منابع /reports
و /forecasts
که قبلا ذکر شد.
فرض کنید میخواهید زمانی که تماسها به منابع /reports
یا /forecasts
میآیند، Edge "کاری انجام دهد". در این مرحله شما به Edge نمیگویید که چه کاری انجام دهد، فقط باید به تماسهای آن منابع گوش دهد. شما این کار را با شرایط انجام می دهید. در پروکسی Edge API خود، میتوانید جریانهای شرطی برای /reports
و /forecasts
ایجاد کنید. برای اهداف مفهومی، پروکسی API XML زیر نشان میدهد که این شرایط چگونه ممکن است به نظر برسند.
<Flows> <Flow name="reports"> <Description/> <Request/> <Response/> <Condition>(proxy.pathsuffix MatchesPath "/reports") and (request.verb = "GET")</Condition> </Flow> <Flow name="forecasts"> <Description/> <Request/> <Response/> <Condition>(proxy.pathsuffix MatchesPath "/forecasts") and (request.verb = "GET")</Condition> </Flow> </Flows>
این شرایط میگوید: «وقتی درخواست GET با /reports
و /forecasts
در URL وارد میشود، Edge هر کاری را که شما (توسعهدهنده API) به آن بگویید، از طریق خطمشیهایی که به آن جریانها متصل میکنید، انجام میدهد.
حالا در اینجا مثالی است که به Edge می گوییم در صورت تحقق یک شرط چه کاری انجام دهد. در XML پروکسی API زیر، وقتی یک درخواست GET به https://yourorg-test.apigee.net/mygreatweatherforecast/reports
ارسال می شود، Edge خط مشی "XML-to-JSON-1" را در پاسخ اجرا می کند.
<Flows> <Flow name="reports"> <Description/> <Request/> <Response> <Step> <Name>XML-to-JSON-1</Name> </Step> </Response> <Condition>(proxy.pathsuffix MatchesPath "/reports") and (request.verb = "GET")</Condition> </Flow>
علاوه بر آن جریانهای شرطی اختیاری، هر پروکسی API نیز دارای دو جریان پیشفرض است: یک <PreFlow>
که قبل از جریانهای شرطی شما اجرا میشود، و یک <PostFlow>
که پس از جریانهای شرطی شما اجرا میشود. هنگامی که هر تماسی با یک پروکسی API برقرار می شود، این موارد برای اجرای خط مشی ها مفید هستند. برای مثال، اگر میخواهید کلید API برنامهای را با هر تماس تأیید کنید، بدون توجه به منبع پشتیبان مورد دسترسی، میتوانید یک خطمشی Verify API Key را در <PreFlow>
قرار دهید. برای اطلاعات بیشتر در مورد جریان، به پیکربندی جریان ها مراجعه کنید.
جریان های مشروط را برای منابع پشتیبان ایجاد کنید
تعریف جریان های شرطی به منابع پشتیبان در یک پراکسی API کاملا اختیاری است. با این حال، آن جریان های مشروط به شما توانایی اعمال مدیریت و نظارت دقیق را می دهد.
شما قادر خواهید بود:
- مدیریت را به گونه ای اعمال کنید که معنایی مدل API شما را منعکس کند
- اعمال خطمشیها و رفتار اسکریپتشده برای مسیرهای منابع فردی (URI)
- معیارهای دقیق را برای سرویس های Analytics جمع آوری کنید
به عنوان مثال، تصور کنید که باید انواع مختلفی از منطق را در backend/developers خود در منابع /apps اعمال کنید.
برای انجام این کار، دو جریان مشروط را در پروکسی API خود اضافه میکنید: /developers
و /apps
.
در نمای توسعه از پنجره ناوبر ویرایشگر پروکسی API، روی نماد + در کنار پیشفرض در نقاط پایانی پروکسی کلیک کنید.
در پنجره "جریان مشروط جدید"، تنظیمات کلیدی زیر را وارد کنید:
- نام جریان : توسعه دهندگان
- نوع شرایط : مسیر
- مسیر : / توسعه دهندگان
اگر تماسی به پروکسی با /developers در انتهای URI ارسال شود، این شرط فعال میشود (و خطمشیها اجرا میشوند).
اکنون یک جریان شرطی برای /apps اضافه کنید، و فرض کنید میخواهید شرط در URI و فعل POST در یک درخواست فعال شود. پیکربندی شامل تنظیم موارد زیر است:
- نام جریان : برنامه ها
- نوع شرط : مسیر و فعل
- مسیر : /apps
- فعل : POST
اگر تماسی با /apps در انتهای URI و یک فعل POST به پروکسی ارسال شود، شرط فعال میشود (و خطمشیها اجرا میشوند).
در قسمت Navigator، جریانهای جدیدی را برای برنامهها و توسعهدهندگان خواهید دید.
یکی از جریان ها را برای مشاهده پیکربندی جریان شرطی در نمای کد ویرایشگر پروکسی API انتخاب کنید:
<Flow name="Apps"> <Description>Developer apps registered in Developer Services</Description> <Request/> <Response/> <Condition>(proxy.pathsuffix MatchesPath "/apps") and (request.verb = "POST")</Condition> </Flow>
همانطور که می بینید، منابع API به سادگی جریان های شرطی هستند که مسیر URI درخواست ورودی را ارزیابی می کنند. (متغیر proxy.pathsuffix URI درخواستی را که از BasePath پیکربندی شده در پیکربندی ProxyEndpoint پیروی می کند، شناسایی می کند.)
هر منبع API که تعریف می کنید توسط یک Flow شرطی در پراکسی API پیاده سازی می شود. (به پیکربندی جریان ها مراجعه کنید.)
پس از استقرار پراکسی API در محیط آزمایشی، درخواست زیر را انجام دهید:
http://{org_name}-test.apigee.net/{proxy_path}/apps
باعث می شود شرط به درستی ارزیابی شود و این جریان به همراه هر خط مشی مرتبط اجرا می شود.
شرط مثال زیر از یک عبارت معمولی جاوا برای تشخیص تماس های برقرار شده با منبع /apps
با یا بدون اسلش رو به جلو ( /apps
یا /apps/**
) استفاده می کند:
<Condition>(proxy.pathsuffix JavaRegex "/apps(/?)") and (request.verb = "POST")</Condition>
برای اطلاعات بیشتر در مورد این نوع شرایط، به نحوه مطابقت بدون توجه به ... در انجمن Apigee مراجعه کنید.
مدلسازی URIهای سلسله مراتبی
در برخی موارد، منابع API سلسله مراتبی خواهید داشت. به عنوان مثال، Developer Services API روشی را برای فهرست کردن همه برنامههایی که به یک توسعهدهنده تعلق دارند، ارائه میکند. مسیر URI:
/developers/{developer_email}/apps
ممکن است منابعی داشته باشید که در آنها یک شناسه منحصربهفرد برای هر موجودیت در یک مجموعه ایجاد میشود که گاهی اوقات به صورت زیر مشروح میشود:
/genus/:id/species
این مسیر به طور مساوی برای دو URI زیر اعمال می شود:
/genus/18904/species /genus/17908/species
برای نشان دادن این ساختار در یک منبع API، می توانید از حروف عام استفاده کنید. به عنوان مثال:
/developers/*/apps
/developers/*example.com/apps
/genus/*/species
این URI های سلسله مراتبی را به عنوان منابع API به طور مناسب حل می کند.
در برخی موارد، به ویژه برای APIهای عمیقاً سلسله مراتبی، ممکن است بخواهید همه موارد زیر یک قطعه URI خاص را حل کنید. برای انجام این کار، از علامت علامت دو ستاره در تعریف منبع خود استفاده کنید. به عنوان مثال، اگر منبع API زیر را تعریف کنید:/developers/**
آن منبع API مسیرهای URI زیر را حل می کند:
/developers/{developer_email}/apps /developers/{developer_email}/keys /developers/{developer_email}/apps/{app_id}/keys
در اینجا شرایط جریان شرطی در تعریف پراکسی API به نظر می رسد:
<Condition>(proxy.pathsuffix MatchesPath "/developers/**") and (request.verb = "POST")</Condition>
نمونه های بیشتر
شرایط متصل به RouteRule
<RouteRule name="default">
<!--this routing executes if the header indicates that this is an XML call. If true, the call is routed to the endpoint XMLTargetEndpoint-->
<Condition>request.header.content-type = "text/xml"</Condition>
<TargetEndpoint>XmlTargetEndpoint</TargetEndpoint>
</RouteRule>
شرایط پیوست به یک سیاست
<Step> <!--the policy MaintenancePolicy only executes if the response status code is exactly 503--> <Condition>response.status.code = 503</Condition> <Name>MaintenancePolicy</Name> </Step>
جریان مشروط
<!-- this entire flow is executed only if the request verb is a GET--> <Flow name="GetRequests"> <Condition>request.verb="GET"</Condition> <Request> <Step> <!-- this policy only executes if request path includes a term like statues--> <Condition>request.path ~ "/statuses/**"</Condition> <Name>StatusesRequestPolicy</Name> </Step> </Request> <Response> <Step> <!-- this condition has multiple expressions. The policy executes if the response code status is exactly 503 or 400--> <Condition>(response.status.code = 503) or (response.status.code = 400)</Condition> <Name>MaintenancePolicy</Name> </Step> </Response> </Flow>
نمونه عملگرها در شرایط
در اینجا چند نمونه از عملگرهای مورد استفاده برای ایجاد شرایط آورده شده است:
-
request.header.content-type = "text/xml"
-
request.header.content-length < 4096 && request.verb = "PUT"
-
response.status.code = 404 || response.status.code = 500
-
request.uri MatchesPath "/*/statuses/**"
-
request.queryparam.q0 NotEquals 10
یک مثال عملی: "/" را در انتهای مسیر نادیده بگیرید
توسعه دهندگان لبه معمولاً می خواهند هر دو پسوند مسیر را مدیریت کنند: " /cat
" و " /cat/
". این به این دلیل است که برخی از کاربران یا مشتریان ممکن است API شما را با اسلش اضافی در انتهای مسیر فراخوانی کنند و شما باید بتوانید آن را در دستورات شرطی خود مدیریت کنید. این مورد استفاده دقیق در انجمن Apigee مورد بحث قرار گرفته است .
اگر ترجیح میدهید، میتوانید بدون استفاده از Regex به این هدف دست پیدا کنید:
<PreFlow name="PreFlow"> <Request> <Step> <Condition>((proxy.pathsuffix = "/cat") OR (proxy.pathsuffix = "/cat/")</Condition> <Name>SomePolicy</Name> </Step> </Request> <Response/> </PreFlow>
این گزینه خوبی است. واضح و خواندنی است.
شما می توانید همین کار را با Regex انجام دهید، اما به این صورت. از پرانتزها برای گروه بندی قسمت regex دستور استفاده می شود، اما نیازی به آنها نیست.
<Condition>(proxy.pathsuffix JavaRegex "/cat(/?)"</Condition>
تماس های API:
GET http://artomatic-test.apigee.net/matchtest/cat
or
GET http://artomatic-test.apigee.net/matchtest/cat
/
آیا سیاست اجرا می شود؟ بله. توجه داشته باشید که در یک عبارت منظم، کاراکتر " ?
" به این معنی است: مطابقت با صفر یا یکی از کاراکترهای قبلی. بنابراین، هر دو " /cat
" و " /cat/
" مطابق هستند.
تماس API:
GET http://artomatic-test.apigee.net/matchtest/cat/spotted
آیا سیاست اجرا می شود؟ نه. عبارت منظم با صفر یا تنها یک مورد از کاراکتر قبلی مطابقت دارد و هیچ چیز دیگری مجاز نیست.
تطبیق رشته های دلخواه با JavaRegex
در تمام مثالهای این مبحث، نحوه تطبیق یکی از متغیرهای جریان داخلی را نشان میدهیم: proxy.pathsuffix. خوب است بدانید که میتوانید تطبیق الگو را روی هر رشته یا متغیر جریان دلخواه انجام دهید، خواه یک متغیر جریان داخلی مانند پسوند proxy.path باشد یا نباشد.
برای مثال، اگر شرطی دارید که یک رشته دلخواه را آزمایش می کند، شاید رشته ای که در یک بار پشتیبان بازگردانده شده است، یا رشته ای که از جستجوی سرور احراز هویت برگردانده شده است، می توانید از عملگرهای منطبق برای آزمایش آن استفاده کنید. اگر از JavaRegex استفاده می کنید، عبارت منظم با کل رشته موضوع مقایسه می شود. اگر موضوع "abc" و عبارت منظم "[az] باشد، مطابقت ندارد، زیرا "[az]" دقیقاً با یک کاراکتر آلفا مطابقت دارد. عبارت "[az]+" مانند "[az]*" و "[az]{3} کار می کند.
بیایید به یک مثال عینی نگاه کنیم. فرض کنید سرور احراز هویت فهرستی از نقشها را بهصورت رشتهای با کاما برمیگرداند: «ویرایشگر، نویسنده، مهمان».
برای آزمایش حضور نقش ویرایشگر، این ساختار کار نخواهد کرد، زیرا "ویرایشگر" تنها بخشی از کل رشته است.
<Condition>returned_roles ~~ "editor"</Condition>
با این حال، این ساخت و ساز کار خواهد کرد:
<Condition>returned_roles ~~ ".*\beditor\b.*")</Condition>
این کار به این دلیل کار می کند که شکست های کلمه و هر قسمت دیگر رشته را با پیشوند و پسوند .* در نظر می گیرد.
در این مثال، شما همچنین می توانید برای "ویرایشگر" با عملگر Matches تست کنید:
<Condition>returned_roles ~~ "*editor*")</Condition>
با این حال، در مواردی که به دقت بیشتری نیاز دارید، JavaRegex اغلب انتخاب بهتری است.
فرار از دو نقل قول در عبارات JavaRegex
دستور Condition به یک عبارت JavaRegex نیاز دارد که در دو گیومه پیچیده شود. بنابراین، اگر یک عبارت Regex دارید که شامل نقل قولهای دوگانه است، به روشی جایگزین برای مطابقت با آنها نیاز دارید. پاسخ یونیکد است. به عنوان مثال، فرض کنید در یک هدر که شامل نقل قول های دوگانه است، عبور می کنید، مانند موارد زیر:-H 'content-type:multipart/related; type="application/xop+xml"'اگر سعی کنید آن هدر را در یک شرط Regex مطابقت دهید، یک خطای Invalid Condition دریافت خواهید کرد زیرا عبارت شامل نقل قول های دوگانه است:
request.header.Content-Type ~~ "(multipart\/related)(; *type="application\/xop\+xml\")"راه حل این است که به جای گیومه های دوگانه مبتنی بر ASCII با معادل یونیکد آنها،
\u0022
. به عنوان مثال، عبارت زیر معتبر است و نتیجه مورد انتظار را ایجاد می کند:request.header.Content-Type ~~ "(multipart\/related)(; *type=\u0022application\/xop\+xml\u0022)"