Have a go at some error handling

Still not happy with how this is all fitting together. But good enough
for now
This commit is contained in:
Toby Zerner 2015-02-26 12:48:23 +10:30
parent b9c09dc37f
commit a007f4c402
8 changed files with 118 additions and 50 deletions

View File

@ -25,12 +25,17 @@ export default JsonApiAdapter.extend({
// If it's a server error, show an alert message. The alerts controller // If it's a server error, show an alert message. The alerts controller
// has been injected into this adapter. // has been injected into this adapter.
if (errors instanceof JsonApiAdapter.ServerError) { if (errors instanceof JsonApiAdapter.ServerError) {
var message = AlertMessage.create({ var message;
if (errors.status === 401) {
message = 'You don\'t have permission to do this.';
} else {
message = errors.message;
}
var alert = AlertMessage.create({
type: 'warning', type: 'warning',
message: 'Something went wrong: '+errors.message.errors[0].code message: message
}); });
this.get('alerts').send('alert', message); this.get('alerts').send('alert', alert);
return;
} }
return errors; return errors;

View File

@ -76,6 +76,7 @@ export default Ember.Controller.extend(Ember.Evented, UseComposerMixin, {
reply: function() { reply: function() {
var discussion = this.get('model'); var discussion = this.get('model');
var controller = this; var controller = this;
if (this.get('session.isAuthenticated')) {
this.showComposer(function() { this.showComposer(function() {
return ComposerReply.create({ return ComposerReply.create({
user: controller.get('session.user'), user: controller.get('session.user'),
@ -85,6 +86,9 @@ export default Ember.Controller.extend(Ember.Evented, UseComposerMixin, {
} }
}); });
}); });
} else {
this.send('signup');
}
}, },
// This action is called when the start position of the discussion // This action is called when the start position of the discussion

View File

@ -24,8 +24,10 @@ export default Ember.Controller.extend(UseComposer, Paneable, {
var controller = this; var controller = this;
return this.saveAndDismissComposer(discussion).then(function(discussion) { return this.saveAndDismissComposer(discussion).then(function(discussion) {
if (discussion) {
controller.get('index').send('loadResults'); controller.get('index').send('loadResults');
controller.transitionToRoute('discussion', discussion); controller.transitionToRoute('discussion', discussion);
}
}); });
}, },
@ -46,6 +48,7 @@ export default Ember.Controller.extend(UseComposer, Paneable, {
newDiscussion: function() { newDiscussion: function() {
var controller = this; var controller = this;
if (this.get('session.isAuthenticated')) {
this.showComposer(function() { this.showComposer(function() {
return ComposerDiscussion.create({ return ComposerDiscussion.create({
user: controller.get('session.user'), user: controller.get('session.user'),
@ -54,6 +57,9 @@ export default Ember.Controller.extend(UseComposer, Paneable, {
} }
}); });
}); });
} else {
this.send('signup');
}
}, },
discussionRemoved: function(discussion) { discussionRemoved: function(discussion) {

View File

@ -62,7 +62,7 @@ export default Ember.View.extend(HasItemLists, {
populateControls: function(items) { populateControls: function(items) {
var view = this; var view = this;
this.addActionItem(items, 'reply', 'Reply', 'reply', 'discussion.canReply', function() { this.addActionItem(items, 'reply', 'Reply', 'reply', null, function() {
view.get('streamContent').send('goToLast'); view.get('streamContent').send('goToLast');
view.get('controller').send('reply'); view.get('controller').send('reply');
}); });

View File

@ -23,8 +23,6 @@ abstract class BaseAction extends Action
public function handle(Request $request, $routeParams = []) public function handle(Request $request, $routeParams = [])
{ {
$this->registerErrorHandlers(); // @todo convert to middleware and add to route group?
$params = array_merge($request->all(), $routeParams); $params = array_merge($request->all(), $routeParams);
return $this->call($params); return $this->call($params);
@ -87,31 +85,6 @@ abstract class BaseAction extends Action
return $this->respondWithArray($document->toArray(), $statusCode, $headers); return $this->respondWithArray($document->toArray(), $statusCode, $headers);
} }
protected function registerErrorHandlers()
{
// if (! Config::get('app.debug')) {
// App::error(function ($exception, $code) {
// return $this->respondWithError('ApplicationError', $code);
// });
// }
// App::error(function (ModelNotFoundException $exception) {
// return $this->respondWithError('ResourceNotFound', 404);
// });
// App::error(function (ValidationFailureException $exception) {
// $errors = [];
// foreach ($exception->getErrors()->getMessages() as $field => $messages) {
// $errors[] = [
// 'code' => 'ValidationFailure',
// 'detail' => implode("\n", $messages),
// 'path' => $field
// ];
// }
// return $this->respondWithErrors($errors, 422);
// });
}
protected function respondWithErrors($errors, $httpCode = 500) protected function respondWithErrors($errors, $httpCode = 500)
{ {
return Response::json(['errors' => $errors], $httpCode); return Response::json(['errors' => $errors], $httpCode);

View File

@ -29,7 +29,7 @@ class TokenAction extends BaseAction
$user = $this->users->findByIdentification($identification); $user = $this->users->findByIdentification($identification);
if (! $user || ! $user->checkPassword($password)) { if (! $user || ! $user->checkPassword($password)) {
return $this->respondWithError('invalidLogin', 401); return $this->respondWithError('invalidCredentials', 401);
} }
$command = new GenerateAccessTokenCommand($user->id); $command = new GenerateAccessTokenCommand($user->id);

View File

@ -12,6 +12,11 @@ class ApiServiceProvider extends ServiceProvider
*/ */
public function boot() public function boot()
{ {
$this->app->singleton(
'Illuminate\Contracts\Debug\ExceptionHandler',
'Flarum\Api\ExceptionHandler'
);
include __DIR__.'/routes.php'; include __DIR__.'/routes.php';
BaseSerializer::setActor($this->app['Flarum\Core\Support\Actor']); BaseSerializer::setActor($this->app['Flarum\Core\Support\Actor']);

View File

@ -0,0 +1,75 @@
<?php namespace Flarum\Api;
use Exception;
use Illuminate\Foundation\Exceptions\Handler;
use Illuminate\Http\Response;
use Illuminate\Http\JsonResponse;
use Illuminate\Database\Eloquent\ModelNotFoundException;
use Flarum\Core\Exceptions\ValidationFailureException;
use Flarum\Core\Exceptions\PermissionDeniedException;
use Symfony\Component\HttpKernel\Exception\HttpException;
use Config;
class ExceptionHandler extends Handler
{
/**
* A list of the exception types that should not be reported.
*
* @var array
*/
protected $dontReport = [
'Symfony\Component\HttpKernel\Exception\HttpException'
];
/**
* Render an exception into an HTTP response.
*
* @param \Illuminate\Http\Request $request
* @param \Exception $e
* @return \Illuminate\Http\Response
*/
public function render($request, Exception $e)
{
if ($request->is('api/*')) {
if ($e instanceof ValidationFailureException) {
return $this->renderValidationException($e);
}
if ($e instanceof PermissionDeniedException) {
return new Response(null, 401);
}
$error = [];
if (Config::get('app.debug')) {
$error['code'] = (new \ReflectionClass($e))->getShortName();
}
if ($detail = $e->getMessage()) {
$error['detail'] = $detail;
}
$statusCode = $e instanceof HttpException ? $e->getStatusCode() : 500;
if (count($error)) {
return $this->renderErrors([$error], $statusCode);
} else {
return new Response(null, $statusCode);
}
}
return parent::render($request, $e);
}
protected function renderErrors($errors, $httpCode = 500)
{
return new JsonResponse(['errors' => $errors], $httpCode);
}
protected function renderValidationException(ValidationFailureException $e)
{
$errors = [];
foreach ($e->getErrors()->getMessages() as $field => $messages) {
$errors[] = [
'detail' => implode("\n", $messages),
'path' => $field
];
}
return $this->renderErrors($errors, 422);
}
}