Şimdiye kadar src/heroes/Heroes.tsx dosyasında bir json dosyasını içe aktarıyorduk, hatırlayın import heroes from './heroes.json'. Şimdi bir arka uç sunucumuz olduğuna göre, verileri ağdan alabiliriz.
Kent C. Dodds'ın Epic React'inden:
"HTTP istekleri, uygulamalarda yapmamız gereken diğer yaygın yan etkilerdir. Bu, oluşturulan DOM'a uyguladığımız yan etkilerle ya da localStorage gibi tarayıcı API'leriyle etkileşim kurarken olduğu gibi farklı değildir. Tüm bu durumlarda, bunları bir useEffect kancası geri çağırma işlevi içinde yaparız. Bu kancı, belirli değişiklikler meydana geldiğinde, bu değişikliklere dayalı olarak yan etkileri uygularız."
HeroListi yüklerken, uygulamamızın arka uca bir GET isteği yapması gerekmektedir. Bunun için başarısız bir e2e testi yazalım (Kırmızı 1). Şimdilik dosyaya istediğimiz ismi verebiliriz. HeroListi yüklüyoruz, ancak şu anda sunucuya yapılan GET istekleri yok.
axios kullanarak yerleşik fetch API'si yerine başka bir seçenek kullanacağız. yarn add axios. Bir useEffect kancasında axios.get kullanarak sunucumuza bir GET isteği yapın (Yeşil 1). useEffect, bileşenin bağını kaldıran bir temizleme işlevi alır.
Test başarılı olur, ancak konsola baktığımızda bileşenin bağlandığını, bağının kaldırıldığını ve tekrar bağlandığını görürüz. Bu arada, API'ye 2 çağrı yapılır. İlk çağrıda yanıt gövdesinde hiçbir şey yoktur, ikincisi ise kahramanları alır. Burada useEffect bağımlılık dizisi ile ilgili bir yan not düşelim ve konuya daha sonra geri dönelim.
useEffect(fn, [a, b, c]) -> a, b veya c değiştiğinde etkiyi çalıştır
useEffect(fn, [a]) -> a değiştiğinde etkiyi çalıştır
useEffect(fn, []) -> etkiyi çalıştır... hiçbir şey değişmediğinde, bu yüzden sadece bir kez çalışır
useEffect(fn) -> her render'da etkiyi çalıştır
Sabit kodlanmış json verilerini kaldırma
heroes verilerini db.json dosyasından içe aktarıyoruz. Şimdi bunları ağdan almanın zamanı. İçe aktarmayı devre dışı bıraktığımızda, değişken artık mevcut olmadığı için bileşende tür hataları alırız, ayrıca listede hiç kahraman olmadığı için test başarısız olur (Kırmızı 2).
useEffect ile bazı veriler alıyoruz, ancak bu verileri bileşendeki bir state değişkeninde saklamamız gerekiyor. const [heroes, setHeroes] = useState([]) şeklinde useState kullanacağız ve ağdan aldığımız kahramanları ayarlayacağız.
Bu işe yarayacaktır, ancak testi açık bırakırsak, sunucuya tekrarlanan GET istekleri yapıldığını ve bileşenin sürekli bağlandığını göreceğiz. Bileşenin render edildiğinde etkinin sadece bir kez gerçekleşmesi için boş bir useEffect bağımlılık dizisi kullanabiliriz. Kısaltmak adına, işte değiştirilen kod (Yeşil 2):
useEffect(() => {console.log("mounting");console.log("heroes is :", heroes);getData().then((data) => {setHeroes(data); });return () =>console.log("unmounting");}, []); // empty array to have the effect occur only once
Biraz daha fazla düzenleme yaparak axios hata mesajlarına destek ekleyebilir ve maliyetli axios.get işlemini useCallBack içinde sarabiliriz. Neden useCallback? Kısacası, özel fonksiyonlar her render işlemi sırasında tanımlanır ve özellikle ağ durumu aynı olduğunda maliyetli olabilir. useCallback, değerlerin yeniden tanımlanmasını veya yeniden hesaplanmasını önleyerek böyle maliyetli işlemleri hafızaya almayı sağlar. İmza şu şekildedir: useCallBack(updaterFn, [dependencies]) (Düzenleme 2).
Heroes bileşeninin tasarımını e2e test ile yönlendirdik. Artık useEffect kullanarak bileşen bir axios isteği yapacak. Bileşen testi Heroes.cy.tsx çalıştırılırken ağ çağrısının gerçekleştiğini ve başarısız olduğunu görebiliriz.
Bileşenin bunu render edebilmesi için ağı bazı verilerle taklit etmemiz gerekiyor. Cypress fixtures'daki heroes.json dosyasından taklit verilerle yanıt verecek şekilde src/heroes/Heroes.cy.tsx ve src/App.cy,tsx dosyalarına bir cy.intercept ekleyin. Intercept, http://localhost:4000/api/heroes adresine yapılan tüm GET isteklerinin Cypress fixtures'daki heroes.json dosyasından gelen taklit verilerle yanıtlanmasını sağlayacaktır.
// src/heroes/Heroes.cy.tsximport Heroes from"./Heroes";import { BrowserRouter } from"react-router-dom";import"../styles.scss";describe("Heroes", () => {beforeEach(() => {cy.intercept("GET","http://localhost:4000/api/heroes", { fixture:"heroes.json", }).as("getHeroes"); });it("should display the hero list on render, and go through hero add & refresh flow", () => {cy.mount( <BrowserRouter> <Heroes /> </BrowserRouter> );cy.wait("@getHeroes");cy.getByCy("list-header").should("be.visible");cy.getByCy("hero-list").should("be.visible");cy.getByCy("add-button").click();cy.location("pathname").should("eq","/heroes/add-hero");cy.getByCy("refresh-button").click();cy.location("pathname").should("eq","/heroes"); });constinvokeHeroDelete= () => {cy.getByCy("delete-button").first().click();cy.getByCy("modal-yes-no").should("be.visible"); };it("should go through the modal flow", () => {cy.window().its("console").then((console) =>cy.spy(console,"log").as("log"));cy.mount( <BrowserRouter> <Heroes /> </BrowserRouter> );cy.getByCy("modal-yes-no").should("not.exist");cy.log("do not delete flow");invokeHeroDelete();cy.getByCy("button-no").click();cy.getByCy("modal-yes-no").should("not.exist");cy.log("delete flow");invokeHeroDelete();cy.getByCy("button-yes").click();cy.getByCy("modal-yes-no").should("not.exist");cy.get("@log").should("have.been.calledWith","handleDeleteFromModal"); });});
Ayrıca, App.cy.tsx'i yansıtan RTL birim testi src/App.test.tsx'i de güncellememiz gerekiyor. Birim testini çalıştırırken başarısız olmaz, ancak hata birleştirme engelleyicisi olacaktır. Sorunun ne olduğunu sadece daha önce gördüğümüz için veya bileşen test çalıştırıcısında gerçek tarayıcıyı kullanarak bileşenin ağ çağrısının gerçekleştiğini gördüğümüz için biliyoruz. Cypress ile bileşen testi, gerçek tarayıcıyı kullanarak Jest / RTL kullanarak yapılması daha zor olan uygulamadaki sorunları teşhis etmeye yardımcı olabilir.
RTL'de cy.intercept'in eşdeğeri msw 'dir. yarn add -D msw ile yükleyin. Dosyayı şu şekilde değiştirin:
Sonunda, tüm e2e testlerinde, URL'yi ziyaret ettikten sonra ağ verilerini beklememiz gerekiyor. Sayfanın stabil olduğundan ve ağ verilerini yüklediğinden emin olmak için, create-hero.cy.tsx, edit-hero.cy.tsx ve network.cy.tsx dosyalarında, cy.visit örneklerini bir kesme ve bekleme ile sarın:
ListHeader bileşeninin + düğmesini kullanarak, herhangi bir ekrandan bir kahraman ekleyebiliriz. Mevcut e2e testimiz, kahraman listesinden veya doğrudan URL'ye yönlendirerek HeroDetails'e ulaşır. Ne yazık ki, kahramanı düzenleme yoluyla da HeroDetails'e ulaşabiliriz, bu da kahraman verileri ile HeroDetais'in başka bir oluşturmasıdır. + düğmesine tıkladığında görüntülenen Kimlik alanının ve ad ve açıklama alanlarındaki verilerin temizlenmesi gereken bu akış ilginçtir. Bunu test etmek için bir test yazalım (Kırmızı 3).
// cypress/e2e/edit-hero.cy.tsdescribe("Edit hero", () => {beforeEach(() => {cy.intercept("GET","http://localhost:4000/api/heroes").as("getHeroes");cy.visit("/");cy.wait("@getHeroes");cy.location("pathname").should("eq","/heroes"); });it("should go through the cancel flow", () => {cy.fixture("heroes").then((heroes) => {cy.getByCy("edit-button").eq(0).click();cy.location("pathname").should("include",`/heroes/edit-hero/${heroes[0].id}` );cy.getByCy("hero-detail").should("be.visible");cy.getByCy("input-detail-id").should("be.visible");cy.findByDisplayValue(heroes[0].id).should("be.visible");cy.findByDisplayValue(heroes[0].name).should("be.visible");cy.findByDisplayValue(heroes[0].description).should("be.visible");cy.getByCy("cancel-button").click();cy.location("pathname").should("eq","/heroes");cy.getByCy("hero-list").should("be.visible"); }); });it("should go through the cancel flow for another hero", () => {cy.fixture("heroes").then((heroes) => {cy.getByCy("edit-button").eq(1).click();cy.location("pathname").should("include",`/heroes/edit-hero/${heroes[1].id}` );cy.getByCy("hero-detail").should("be.visible");cy.getByCy("input-detail-id").should("be.visible");cy.findByDisplayValue(heroes[1].id).should("be.visible");cy.findByDisplayValue(heroes[1].name).should("be.visible");cy.findByDisplayValue(heroes[1].description).should("be.visible");cy.getByCy("cancel-button").click();cy.location("pathname").should("eq","/heroes");cy.getByCy("hero-list").should("be.visible"); }); });it("should navigate to add from an existing hero", () => {cy.fixture("heroes").then((heroes) => {cy.getByCy("edit-button").eq(1).click();cy.getByCy("add-button").click();cy.getByCy("input-detail-id").should("not.exist");cy.findByDisplayValue(heroes[1].name).should("not.exist");cy.findByDisplayValue(heroes[1].description).should("not.exist"); }); });});
Test başarısız olur. Koşullu işleme çalışıyor, ancak InputDetail bileşeninin ( HeroDetail'in alt bileşeni) durumu devredilir.
Değeri alıp sadece görüntülemek yerine, InputDetail'i durumun farkında kılmalıyız. Bunu, durumu en alakalı olduğu yerde, bileşenin kendisinde yöneterek ve useState ve useEffect kombinasyonunu kullanarak başarabiliriz. shownValue adında bir değişken ve onun için bir ayarlayıcı bulunur. Bileşen bağlandığında, useEffect'i değeri ayarlamak için kullanırız. Ayrıca, değer için bir bağımlılık dizisi belirtiriz (Yeşil 3).
Benzer şekilde yönlendirmeye, uygulamadaki endişelerimiz daha üst düzey olduğunda, yani durum yönetimi ve akışlar söz konusu olduğunda, e2e testler, bileşen testleriyle test edemeyeceğimiz hataları yakalamada etkili olabilir. E2e test şimdi çalışıyor ve bileşen testi geriye dönük güvence sağlar. onChange şimdi iki kez vs üç kez çağrılır ve bu güncellemenin tek farkıdır.
// src/components/InputDetail.cy.tsximport InputDetail from"./InputDetail";import"../styles.scss";describe("InputDetail", () => {constplaceholder="Aslaug";constname="name";constvalue="some value";constnewValue="42";it("should allow the input field to be modified", () => {cy.mount( <InputDetailname={name}value={value}placeholder={placeholder}onChange={cy.stub().as("onChange")} /> );cy.contains("label", name);cy.findByPlaceholderText(placeholder).clear().type(newValue);cy.findByDisplayValue(newValue);cy.get("@onChange").its("callCount").should("eq",newValue.length); });it("should not allow the input field to be modified", () => {cy.mount( <InputDetailname={name}value={value}placeholder={placeholder}readOnly={true} /> );cy.contains("label", name);cy.findByPlaceholderText(placeholder).should("have.value", value).and("have.attr","readOnly"); });});
Yeniden Düzenleme
Ortam değişkenleri
Tüm http://localhost:4000/api referanslarını ortam değişkeniyle yeniden düzenleme zamanı geldi. Create React App (CRA), dotenv paketiyle zaten kurulmuş durumda. Tek gereksinim, değişken adlarının REACT_APP_ ile başlamasıdır. Hemen API URL'si ile bir .env dosyası oluşturabiliriz.
.env dosyasının eşdeğeri, ./cypress.config.js dosyasındaki env özelliğidir. Bu özellik, e2e, component veya her ikisine özgü olabilir ve özelliğin yerleştirildiği yere bağlıdır.
Bileşen ve e2e testlerindeki http://localhost:4000/api dizesi örneklerini, ${Cypress.env('API_URL')} şablon dizesiyle değiştirin. Değişiklik yapılması gereken dosyalar şunlardır:
cypress/e2e/create-hero.cy.ts
cypress/e2e/edit-hero.cy.ts
cypress/e2e/network-hero.cy.ts
cypress/support/commands.ts
src/App.cy.tsx
src/components/InputDetail.cy.tsx
src/heroes/Heroes.cy.tsx
Özel kanca useAxios
HTTP mantığının 20-30 satırını kendi kanca içine çıkarabilir ve ardından Heroes bileşeninde bu kanca kullanabiliriz. Kancamız argüman olarak bir rota kabul eder, veri, durum ve hata nesnesini döndürür. Ayrıca useEffect temizleme ile ilgili kaygıları ele alır. Detayları yorumlarda ve gelecek bölümlerde daha iyi bir çözüm kullanarak ele alacağız.
// src/hooks/useAxios.tsimport { useCallback, useEffect, useState } from"react";import axios from"axios";constgetItem= (route:string) =>axios({ method:"GET", baseURL:`${process.env.REACT_APP_API_URL}/${route}`, }).then((res) =>res.data).catch((err) => {throwError(`There was a problem fetching data: ${err}`); });/** Takes a url, returns an object of data, status & error */exportdefaultfunctionuseAxios(url:string) {const [data,setData] =useState();const [error,setError] =useState(null);const [status,setStatus] =useState("idle");constgetItemCb=useCallback((route:string) => {returngetItem(route); }, []);// When fetching data within a call to useEffect,// combine a local variable and the cleanup function// in order to match a data request with its response:// If the component re-renders, the cleanup function for the previous render// will set the previous render’s doUpdate variable to false,// preventing the previous then method callback from performing updates with stale data.// eslint-disable-next-line @typescript-eslint/ban-ts-comment// @ts-ignoreuseEffect(() => {let doUpdate =true;setData(undefined);setError(null);setStatus("loading");getItemCb(url).then((data) => {if (doUpdate) {setData(data);setStatus("success"); } }).catch((error) => {if (doUpdate) {setError(error);setStatus("error"); } });return () => (doUpdate =false); }, [url]);return { data, status, error };}
Heroes bileşeninde artık useState kullanmamıza gerek yok çünkü verileri useAxios ile alıyoruz. Sadece data değişkenini yeniden adlandırıp heroes olarak başlatmalıyız.
Başarısız bir e2e testi yazdık, uygulama yüklenirken beklenen bir http GET çağrısına /heroes yolunda casusluk yaptık (Kırmızı 1).
Verileri /heroes rotasına yönelik bir axios.get çağrısı yaparak alan useEffect ekledik (Yeşil 1).
Verileri almak için json dosyası içe aktarmasını kaldırdık, useState kullandık ve veri dizisini useEffect içinde ayarladık (Kırmızı 2, Yeşil 2).
Yeniden düzenlemeler:
Http GET etkisinin sadece bir kez gerçekleşmesi için boş bir dizi kullandık. Pahalı işlevleri sarmak için useCallback gösterdik.
Bileşen testlerini ve ağa bağlı birim testini güncelledik, Cypress için cy.intercept ve RTL için msw ile ağı örnekledik.
DOM yerleştikten sonra kullanıcı arayüzü iddialarının başlayabilmesi için e2e testlerini ağı beklemeye güncelledik.
Alternatif bir kahraman ekleme akışını kapsayan yeni bir e2e testi ekledik; kahramanı düzenlemekten kahraman ekleme yoluna geçiş (Kırmızı 3).
Başarısızlığı ele almak için durumu en alakalı olduğu yerde yönettik; InputDetail bileşeni. Sadece useState ve useEffect kullandık (Yeşil 3).
Yeniden düzenlemeler:
Sabit kodlanmış api yolunu bir ortam değişkenine çevirdik.
Bir useAxios kancası kullanarak verileri bileşende soyut bir şekilde sunuyoruz.
Önemli Noktalar
Kent Dodds'tan: "HTTP istekleri, uygulamalarda yapmamız gereken başka yaygın yan etkilerdir. Bu, işlenmiş DOM'a uygulamalar yapmamız gereken yan etkilerle ya da localStorage gibi tarayıcı API'leriyle etkileşimde bulunmamız gereken yan etkilerden farklı değildir. Tüm bu durumlarda, bunları bir useEffect kancası geri çağırma içinde yaparız. Bu kancanın yardımıyla, belirli değişiklikler meydana geldiğinde, bu değişikliklere dayalı yan etkileri uygularız."
Uygulamadan backend'e http çağrıları yapmak için yerleşik fetch API'sini veya axios kullanabiliriz.
useEffect bağımlılık dizisi:
useEffect(fn, [a, b, c]) -> a, b veya c değiştiğinde etkiyi çalıştır
useEffect(fn, [a]) -> a değiştiğinde etkiyi çalıştır
useEffect(fn, []) -> etkiyi çalıştır... hiçbir şey değişmediğinde, bu yüzden sadece bir kez çalışır
useEffect(fn) -> her render işlemde etkiyi çalıştır
Tekrarlanan çağrıları hafızada tutmak için pahalı işlevleri useCallback ile sarın.
useState ve useEffect ile çoğu http durumunu yönetebiliriz, ancak uygulama ölçeklendikçe uygulama büyüyebilir.
Cypress ile bileşen testi yaparak, gerçek tarayıcıyı kullanarak, Jest/RTL kullanarak yapmaktan daha zor olabilecek uygulama sorunlarını teşhis etmeye yardımcı olabilir.
Bileşen testleri ağ çağrıları yapıyorsa, cy.intercept API'si ile ağı örnekleyebiliriz. Jest/RTL için cy.intercept'in zıttı msw 'dir.
Yönlendirmeye benzer şekilde, uygulamayla ilgili endişelerimiz durum yönetimi ve akışlar gibi daha yüksek düzeyde olduğunda, bileşen testleriyle kapsayamayabileceğimiz kenar durumlarını yakalamak için e2e testleri etkilidir.