闄愰€熷彲浠ヤ繚鎶ゅ拰鎻愰珮鍩轰簬API鐨勬湇鍔$殑鍙敤鎬с€傚鏋滄偍姝e湪涓嶢PI瀵硅瘽骞舵敹鍒癏TTP429TooManyRequests鍝嶅簲鐘舵€佷唬鐮侊紝鍒欒鏄庢偍鐨勯€熺巼鍙楀埌闄愬埗銆傝繖鎰忓懗鐫€鎮ㄨ秴鍑轰簡缁欏畾鏃堕棿鍐呭厑璁哥殑璇锋眰鏁伴噺銆傛偍闇€瑕佸仛鐨勫氨鏄斁鎱㈤€熷害锛岀◢绛夌墖鍒伙紝鐒跺悗閲嶈瘯銆備负浠€涔堣闄愰€燂紵褰撴偍鑰冭檻闄愬埗鑷繁鐨勫熀浜嶢PI鐨勬湇鍔℃椂锛屾偍闇€瑕佸湪鐢ㄦ埛浣撻獙銆佸畨鍏ㄦ€у拰鎬ц兘涔嬮棿鍋氬嚭鏉冭 銆傛帶鍒舵暟鎹祦鐨勬渶甯歌鍘熷洜鏄淮鎶ゅ熀浜嶢PI鐨勬湇鍔$殑鍙敤鎬с€備絾涔熸湁瀹夊叏鏂归潰鐨勫ソ澶勶紝鍥犱负鍏ョ珯娴侀噺鐨勪竴娆℃棤鎰忔垨鏈夋剰婵€澧炲彲鑳戒細鍗犵敤瀹濊吹鐨勮祫婧愬苟褰卞搷鍏朵粬鐢ㄦ埛鐨勫彲鐢ㄦ€с€傞€氳繃鎺у埗浼犲叆璇锋眰鐨勯€熺巼锛屾偍鍙互锛氱‘淇濇湇鍔″拰璧勬簮涓嶄細鈥滀笉鍫噸璐熲€濄€傚噺杞绘毚鍔涙敾鍑婚槻姝㈠垎甯冨紡鎷掔粷鏈嶅姟(DDOS)鏀诲嚮濡備綍瀹炴柦閫熺巼闄愬埗锛熼€熺巼闄愬埗鍙互鍦ㄥ鎴风绾у埆銆佸簲鐢ㄧ▼搴忕骇鍒€佸熀纭€鏋舵瀯绾у埆鎴栦袱鑰呬箣闂寸殑浠讳綍浣嶇疆瀹炴柦銆傛湁鍑犵鏂规硶鍙互鎺у埗API鏈嶅姟鐨勫叆绔欐祦閲忥細姣忎釜鐢ㄦ埛锛氳窡韪敤鎴蜂娇鐢ˋPI瀵嗛挜銆佽闂护鐗屾垨IP鍦板潃杩涜鐨勮皟鐢ㄤ粠鍦扮悊涓婅锛氫緥濡傦紝鍦ㄤ竴澶╀腑鐨勯珮宄版椂娈甸檷浣庢瘡涓湴鐞嗗尯鍩熺殑璐圭巼闄愬埗姣忎釜鏈嶅姟鍣細濡傛灉鎮ㄦ湁澶氫釜鏈嶅姟鍣ㄥ鐞嗗鎮ㄧ殑API鐨勪笉鍚岃皟鐢紝鎮ㄥ彲浠ュ璁块棶鏇存槀璐电殑璧勬簮瀹炴柦鏇翠弗鏍肩殑閫熺巼闄愬埗銆傛偍鍙互浣跨敤杩欎簺閫熺巼闄愬埗涓殑浠讳綍涓€涓紙鐢氳嚦缁勫悎锛夈€傛棤璁烘偍閫夋嫨濡備綍瀹炴柦瀹冿紝閫熺巼闄愬埗鐨勭洰鏍囬兘鏄缓绔嬩竴涓鏌ョ偣锛岃妫€鏌ョ偣鎷掔粷鎴栦紶閫掕闂偍鐨勮祫婧愮殑璇锋眰銆傝澶氱紪绋嬭瑷€鍜屾鏋堕兘鏈夊唴缃殑鍑芥暟鎴栦腑闂翠欢鏉ュ疄鐜拌繖涓€鐐癸紝骞朵笖鏈夊悇绉嶉€熺巼闄愬埗绠楁硶鐨勯€夐」銆傝繖鏄娇鐢∟ode鍜孯edis鍒涘缓鎮ㄨ嚜宸辩殑閫熺巼闄愬埗鍣ㄧ殑涓€绉嶆柟娉曪細鍒涘缓涓€涓狽ode搴旂敤绋嬪簭浣跨敤Redis娣诲姞涓€涓€熺巼闄愬埗鍣ㄥ湪Postman涓祴璇曞畠馃捇鏌ョ湅GitHub涓婄殑浠g爜绀轰緥銆傚湪寮€濮嬩箣鍓嶏紝璇风‘淇濇偍鐨勮绠楁満涓婂畨瑁呬簡Node鍜孯edis銆傜1姝ワ細鏋勫缓Node搴旂敤绋嬪簭浠庡懡浠よ璁剧疆鏂扮殑Node搴旂敤绋嬪簭銆傞€氳繃CLI鎻愮ず锛屾垨閫氳繃娣诲姞--yes鏍囧織鎺ュ彈榛樿閫夐」銆傚鏋滃湪椤圭洰璁剧疆鏈熼棿鎺ュ彈浜嗛粯璁ら€夐」锛屽垯$npminit--yes浼氫负鍏ュ彛鐐瑰垱寤轰竴涓悕涓篿ndex.js鐨勬枃浠躲€?touchindex.js瀹夎Expressweb妗嗘灦锛岀劧鍚庡湪index.js涓垵濮嬪寲鏈嶅姟鍣ㄣ€俢onstexpress=require('express')constapp=express()constport=process.env.PORT||3000app.get('/',(req,res)=>res.send('HelloWorld!'))app.listen(port,()=>console.log(`绀轰緥搴旂敤绋嬪簭鍦╤ttp://localhost鐩戝惉:${port}`))浠庡懡浠よ鍚姩鏈嶅姟鍣ㄣ€?nodeindex.js鍥炲埌index.js锛屽垱寤轰竴涓矾鐢憋紝棣栧厛妫€鏌ラ€熺巼闄愬埗锛岀劧鍚庡鏋滅敤鎴锋病鏈夎秴杩囬檺鍒跺垯鍏佽璁块棶璧勬簮銆俛pp.post('/',async(req,res)=>{asyncfunctionisOverLimit(ip){//瀹氫箟}//妫€鏌ラ€熺巼闄愬埗letoverLimit=awaitisOverLimit(req.ip)if(overLimit){res.status(429).send('Toomanyrequests-tryagainlater')return}//鍏佽璁块棶璧勬簮res.send("Accessedthepreciousresources!")})鍦ㄤ笅涓€姝ヤ腑鎴戜滑灏嗗畾涔夐€熺巼闄愬埗鍣ㄥ嚱鏁版槸OverLimit銆傜2姝ワ細浣跨敤Redis娣诲姞閫熺巼闄愬埗鍣≧edis鏄竴涓唴瀛樹腑鐨勯敭鍊兼暟鎹簱锛屽洜姝ゅ畠鍙互闈炲父蹇€熷湴妫€绱㈡暟鎹€備娇鐢≧edis瀹炵幇閫熺巼闄愬埗涔熼潪甯哥畝鍗曘€傚瓨鍌ㄤ竴涓瘑閽ワ紝渚嬪鐢ㄦ埛鐨処P鍦板潃銆傚湪鎸囧畾鐨勬椂闂存鍚庡鍔犱粠璇PExpire璁板綍鍙戝嚭鐨勮皟鐢ㄦ鏁颁笅鍥炬墍绀虹殑閫熺巼闄愬埗绠楁硶鏄竴涓粦鍔ㄧ獥鍙h鏁板櫒鐨勭ず渚嬨€傚鏋滅敤鎴锋彁浜ら€傚害鏁伴噺鐨勮皟鐢紝鎴栬€呴殢鐫€鏃堕棿鐨勬帹绉诲皢瀹冧滑闂撮殧寮€锛屽垯鐢ㄦ埛姘歌繙涓嶄細杈惧埌閫熺巼闄愬埗銆傚湪10绉掔獥鍙e唴瓒呰繃鏈€澶ц姹傜殑鐢ㄦ埛蹇呴』绛夊緟瓒冲鐨勬椂闂存潵鎭㈠浠栦滑鐨勮姹傘€備粠鍛戒护琛屼负Node瀹夎鍚嶄负ioredis鐨凴edis瀹㈡埛绔€?npminstallioredis鍦ㄦ湰鍦板惎鍔≧edis鏈嶅姟鍣ㄣ€?redis-server鐒跺悗鍦╥ndex.js涓姹傚苟鍒濆鍖朢edis瀹㈡埛绔€俢onstredis=require('ioredis')constclient=redis.createClient({port:process.env.REDIS_PORT||6379,host:process.env.REDIS_HOST||'localhost',})client.on('connect',function(){console.log('connected');});瀹氫箟鎴戜滑涓婁竴姝ュ啓鐨刬sOverLimit鍑芥暟锛屾牴鎹甊edis鐨勮繖绉嶆ā寮忥紝鏍规嵁IP淇濆瓨涓€涓鏁板櫒銆俛syncfunctionisOverLimit(ip){letrestry{res=awaitclient.incr(ip)}catch(err){console.error('isOverLimit:couldnotincrementkey')throwerr}console.log(`${ip}鏈夊€硷細${res}`)if(res>10){returntrue}client.expire(ip,10)}杩欐槸閫熺巼鈥嬧€嬮檺鍒跺櫒銆傚綋鐢ㄦ埛璋冪敤API鏃讹紝鎴戜滑妫€鏌edis浠ユ煡鐪嬬敤鎴锋槸鍚﹁秴杩囬檺鍒躲€傚鏋滄槸杩欐牱锛孉PI灏嗙珛鍗宠繑鍥濰TTP429鐘舵€佷唬鐮佸拰娑堟伅Toomanyrequests鈥攖ryagainlater銆傚鏋滅敤鎴峰湪闄愬埗鑼冨洿鍐咃紝鎴戜滑灏嗙户缁墽琛屼笅涓€涓唬鐮佸潡锛屽湪璇ヤ唬鐮佸潡涓垜浠彲浠ュ厑璁歌闂彈淇濇姢鐨勮祫婧愶紙渚嬪鏁版嵁搴擄級銆傚湪閫熺巼闄愬埗妫€鏌ユ湡闂达紝鎴戜滑鍦≧edis涓壘鍒扮敤鎴风殑璁板綍骞跺鍔犲叾璇锋眰璁℃暟锛屽鏋淩edis涓病鏈夎鐢ㄦ埛鐨勮褰曪紝閭d箞鎴戜滑灏嗗垱寤轰竴涓柊璁板綍銆傛渶鍚庯紝姣忔潯璁板綍灏嗗湪鏈€杩戜竴娆℃椿鍔ㄥ悗鐨?0绉掑唴杩囨湡銆傚湪涓嬩竴姝ヤ腑锛岀‘淇濇垜浠殑闄愰€熷櫒姝e父杩愯銆傜3姝ワ細鍦≒ostman涓祴璇曚繚瀛樻洿鏀瑰苟閲嶆柊鍚姩鏈嶅姟鍣ㄣ€傛垜浠皢浣跨敤Postman鍚戞垜浠殑API鏈嶅姟鍣ㄥ彂閫丳OST璇锋眰锛岃鏈嶅姟鍣ㄥ湪鏈湴杩愯鍦╤ttp://localhost:3000銆傜户缁揩閫熻繛缁彂閫佽姹備互杈惧埌鎮ㄧ殑閫熺巼闄愬埗銆傚叧浜庨€熺巼闄愬埗鐨勬渶缁堟兂娉曡繖鏄疦ode鍜孯edis閫熺巼闄愬埗鍣ㄧ殑涓€涓畝鍗曠ず渚嬶紝鑰岃繖浠呬粎鏄釜寮€濮嬨€傛偍鍙互浣跨敤澶氱绛栫暐鍜屽伐鍏锋潵璁捐鍜屽疄鏂介€熺巼闄愬埗銆傚苟涓斿彲浠ラ€氳繃姝ょず渚嬫帰绱㈠叾浠栧寮哄姛鑳斤紝渚嬪锛氬湪鍝嶅簲鏂囨湰涓垨浣滀负Retry-after鏍囧ご锛岃鐢ㄦ埛鐭ラ亾鍦ㄩ噸璇曚箣鍓嶄粬浠簲璇ョ瓑寰呭闀挎椂闂磋褰曡揪鍒伴€熺巼闄愬埗鐨勮姹備互浜嗚В鐢ㄦ埛琛屼负骞惰鍛婅瘯鍥句娇鐢ㄥ叾浠栭€熺巼闄愬埗绠楁硶鎴栧叾浠栦腑闂翠欢鐨勬伓鎰忔敾鍑昏璁颁綇锛屽綋鎮ㄧ爺绌禔PI鑺傛祦鏃讹紝鎮ㄦ槸鍦ㄦ€ц兘銆佸畨鍏ㄦ€у拰鐢ㄦ埛浣撻獙涔嬮棿杩涜鏉冭 銆傝€冭檻鍒拌繖浜涘洜绱狅紝鎮ㄧ悊鎯崇殑閫熺巼闄愬埗瑙e喅鏂规浼氶殢鐫€鏃堕棿鑰屾敼鍙樸€傚井淇℃悳绱€愬墠绔叏鏍堝紑鍙戙€戝叧娉ㄨ繖浣嶆帀澶村彂銆佹憜鍦版憡銆佸崠璐с€佺户缁涔犵殑绋嬪簭鍛樸€傜涓€鏃堕棿闃呰鏈€鏂版枃绔狅紝杩欎袱澶╀細浼樺厛鍙戝竷鏂版枃绔犮€傚叧娉ㄥぇ绀煎寘锛岃浣犵渷閽憋紒
