J'ai deux applications, l'une est une interface frontale et l'autre, l'application Rails-api.
J'ai été heureux d'utiliser isomorphic-fetch jusqu'à ce que je devais envoyer la méthode PATCH au serveur.
Je reçois:
Fetch API cannot load http://localhost:3000/api/v1/tasks. Method patch is not allowed by Access-Control-Allow-Methods in preflight response.
mais la réponse OPTIONS du serveur inclut une méthode PATCH dans une liste de méthodes Access-Control-Allow-Methods:
Voici comment le fetch est implémenté:
const API_URL = 'http://localhost:3000/'
const API_PATH = 'api/v1/'
fetch(API_URL + API_PATH + 'tasks', {
headers: {
'Accept': 'application/json',
'Content-Type': 'application/json'
},
method: 'patch',
body: JSON.stringify( { task: task } )
})
POST, GET, DELETE sont pratiquement identiques et fonctionnent bien.
Une idée de ce qui se passe ici?
METTRE À JOUR:
Le patch de méthode est sensible à la casse:
https://github.com/github/fetch/blob/master/fetch.js#L200
Je ne sais pas si c'est prévu ou s'il y a un bug.
UPDATE 2
Ceci est prévu et le type de méthode PATCH doit être sensible à la casse . Mettre à jour la ligne de la méthode fetch pour:
method: 'PATCH'
résout le problème.
J'ai eu un problème très similaire avec reactJS front end et Rails API utilisant Rack :: Cors, et l'ajout de patch
à la liste des méthodes autorisées a résolu le problème pour moi.
config.middleware.insert_before 0, Rack::Cors do
allow do
origins '*'
resource '*', headers: :any, methods: [:get, :post, :patch, :options]
end
end
J'ai eu cette erreur alors que PATCH
était en majuscule. Je recevais aussi cette erreur avec DELETE
et PUT
aussi. J'ai vérifié les en-têtes de ma fetch
et j'ai vu une méthode OPTIONS
. J'utilisais la isomorphic-fetch
lib ici - https://www.npmjs.com/package/isomorphic-fetch
La solution pour moi a été d’ajouter à ma page PHP:
<?php
header('Access-Control-Allow-Methods: GET, POST, PUT, DELETE, PATCH');
Sans cela, dans Firefox 53, je continuerais à avoir l'erreur javascript:
NetworkError lors d'une tentative d'extraction de ressource.
Le fetch que je faisais était le suivant:
try {
await fetch('https://my.site.com/', {
method: 'PATCH',
headers: { 'Content-Type':'application/x-www-form-urlencoded' },
body: 'id=12&day=1'
});
} catch(ex) {
console.error('ex:', ex);
}
utilisez ce code _method: 'PATCH'
return (
fetch(API_ROOT + route, {
_method: 'PATCH',
crossDomain: true,
xhrFields: {
withCredentials: true
},
headers: {
Accept: 'application/json',
'Content-Type': 'application/json',
'Authorization': ''
},
data: JSON.stringify(data),
credentials: 'include'
})
.then(res => res.json())
.then(res => {
return res
})
.catch(err => console.error(err))
);
Une autre façon est insérer une méthode dans les en-têtes
headers: {
Accept: 'application/json',
'Content-Type': 'application/json',
'_method': 'PATCH',
'Authorization': ''
}
cela vous aide
return (
fetch(API_ROOT + route, {
method: 'POST',
crossDomain: true,
xhrFields: {
withCredentials: true
},
headers: {
Accept: 'application/json',
'Content-Type': 'application/json',
'_method': 'PATCH',
'Authorization': ''
},
data: JSON.stringify(data)
})
.then(res => res.json())
.then(res => {
console.log(res);
return res
})
.catch(err => console.error(err))
);