Thursday, 5 February 2015

Error Logging with ELMAH

If you don't know about ELMAH you should. Here's how to set it up to Log your MVC 4 application errors.

References:

Install the Nuget Package:

Install-Package Elmah.MVC

Set up Database Logging

Download and run the SQL script https://code.google.com/p/elmah/downloads/detail?name=ELMAH-1.2-db-SQLServer.sql

Web.config

Modify your web.config file as follows:

<elmah></elmah> <elmah> <security allowRemoteAccess="true" /> <errorLog type="Elmah.SqlErrorLog, Elmah" connectionStringName="DefaultConnection" /> </elmah>

Under <appSettings>

<add key="elmah.mvc.requiresAuthentication" value="true" /> <add key="elmah.mvc.allowedRoles" value="Administrator" /> <add key="elmah.mvc.allowedUsers" value="Administrator" />

Try it out:

Login to your application as Administrator and Navigate to: {your url}/elmah

Update: Logging Handled Exceptions

By default ELMAH only logs unmanaged exceptions. To Log handled exceptions add the following to your Try Catch Blocks:

E.g.
try{ }

catch (Exception e)
          {
                Elmah.ErrorSignal.FromCurrentContext().Raise(e);
                ViewBag.ErrorMessage = e.ToString();
                return View("Error");
      }

Thursday, 18 December 2014

Create A Dropdown List with A Range of Numeric Values in ASP.NET MVC 4

If you want to have a dropdownlist list containing a range of numerical values here's how to do it:

Source: http://stackoverflow.com/questions/12070363/dropdown-list-asp-mvc

@{ var values = new SelectList(Enumerable.Range(0, 60)); }

@Html.DropDownListFor(model => model.NoOfUserAccounts,values)
@Html.DropDownListFor(model => model.NoOfUserAccounts)s)

Thursday, 27 November 2014

Entity Framework 5 Code First - Customizing Migrations

You want to make changes to your Models and Database, you have auto migration enabled and you run:

update-database -force

you get this error:

'FK_dbo.{TABLENAME}_dbo.{OTHER_TABLENAME}_otherID' is not a constraint.
Could not drop constraint. See previous errors.

You sh*t a brick.

Solution:

Run this:

Add-Migration Update{You Model Name}Class

This creates a file in Migrations containing a Partial Class outlining what changes the Framework is attempting as a "best guess"

Remove any unwanted changes in this file.

E.g DropForeignKey("dbo.{TABLENAME}", "otherID", "dbo.{OTHER TABLE}");

Now run:

Update-Database –TargetMigration:Update{You Model Name}Class

NOW READ THIS, TWICE!

http://msdn.microsoft.com/en-gb/data/jj591621.aspx#customizing

And if none of that works

If you're still getting errors, and trust me, it happens. Then I recommend backing everything up and starting with a clean slate.

I followed the instructions here http://stackoverflow.com/questions/11679385/reset-entity-framework-migrations from Todd. Thank's Todd :)

  1. Delete the migrations folder in your project
  2. Delete the __MigrationHistory table in your database (may be under system tables)
  3. In PM Console run: Enable-Migrations -EnableAutomaticMigrations -Force
  4. In PM Console run: Add-Migration Initial
  5. In PM Console run: update-databse -TargetMigration:Initial

Wednesday, 6 August 2014

Adding a Horizontal Line to PDF similar to hr tag

iTextSharp.text.pdf.draw.LineSeparator line1 = new iTextSharp.text.pdf.draw.LineSeparator(0f, 100f, BaseColor.BLACK, Element.ALIGN_LEFT, 1);
document.Add(new Chunk(line1));

Friday, 25 July 2014

Restrict a Method To Members of Certain Roles

Simple Example

// Only Let People who are in the Administrator or Managers Roles Use this method
   [Authorize(Roles = "Administrator,Managers")]
        public ActionResult Delete(int id = 0)
        {
            Report Report = db.Report.Find(id);
            if (Report == null)
            {
                return HttpNotFound();
            }
            return View(Report );
        }

MVC 4 - Creating PDF Files with iTextSharp

I've found in my latest job that just about every project requires spitting out PDFs in various flavours so here's a simple example:

// First things first, make sure you've got the package installed
// Package Manager
PM> Install-Package iTextSharp

// In your Controller make sure you reference iTextSharp.text and iTextSharp.pdf
using iTextSharp.text;
using iTextSharp.text.pdf;

// Outputs a Simple PDF with the Name and Description of a given Product
    public FileStreamResult ProductInfoPDF(int id)
        {
        Product = context.Products.Find(id);
// if we don't find anything then just quit
            if (Product == null)
            {
                return null;
            }

            MemoryStream workStream = new MemoryStream();
            Document document = new Document();
           // Landscape Layout
            document.SetPageSize(iTextSharp.text.PageSize.A4.Rotate());
            PdfWriter.GetInstance(document, workStream).CloseStream = false;
            document.Open();
            // Heading
            Paragraph pHeading = new Paragraph("Product Information");
            pHeading.Alignment = Element.ALIGN_CENTER;
            pHeading.Font.SetStyle(Font.BOLD);
            pHeading.Font.Size = 12;
            pHeading.SpacingAfter = 18f;
            document.Add(new Paragraph(pHeading));                  
            Paragraph p = new Paragraph("Some background about this product.");
            p.Font.SetStyle(Font.BOLD);
            p.Font.Size = 8;
            document.Add(p);
            // Product Name
            Paragraph pProductName = new Paragraph("Product Name: " + Product.Name);
            pProductName.Font.SetStyle(Font.BOLD);
            pProductName.Font.Size = 8;
            document.Add(pProductName);
            // Description
            document.Add(new Paragraph("\n"));
            document.Add(new Paragraph("\n"));
            Paragraph pDescription = new Paragraph("Description");
            pEquip.Font.SetStyle(Font.BOLD);
            pEquip.Font.Size = 8;
            document.Add(pDescription );        
            document.Close();
            byte[] byteInfo = workStream.ToArray();
            workStream.Write(byteInfo, 0, byteInfo.Length);
            workStream.Position = 0;
            return new FileStreamResult(workStream, "application/pdf") { FileDownloadName = "Product_" + Product.Id.ToString() + ".pdf" };

        }


// Add  a link in your View
  @Html.ActionLink("Print", "ProductInfoPDF","Products" ,new { id = Model.Id }, new { @class = "printLink"})

LINQ Kung Fu - Tips from t'coalface of t'Interwebs

I find myself constantly having to look up how to do things in LINQ that I find straightforward in SQL, so I'm making a note of them here. Hopefully someone else will find these useful.


Select Statements

Get List based on Category

// Get Products in a certain category
// same as SQL: select * from Products where CategoryId = CategoryId
    int CategoryId = 1;
    List<Product> Products = db.Products.Where(x => x.CategoryId == CategoryId).ToList();

Select a Single Value with Where Clause

// get the Price of the latest product entered with the Name: "Box of Blue Sparks"
// SQL Equivalent: select TOP 1 Price from Products where Name = 'Box of Blue 
// Sparks' ORDER BY CreatedDate DESC
string pName = "Box of Blue Sparks";
  var available = db.Products.Where(x => x.Name == pName).OrderByDescending(x => x.CreatedDate).Select(x => x.Price).FirstOrDefault();
                              

Select DISTINCT

// Get a list of Distinct Product Colours
// SQL Equivalent: SELECT DISTINCT(Colour) FROM Products WHERE Price is more then £10
List<string> ProductNames = db.Products.Where(x => x.Price > 10).Select(x => x.Colour).Distinct().ToList();

// Note to the Colonials, that is the correct way to spell COLOUR!

Select a Single Record

// Get the first product priced over £100
// SQL Equivalant: SELECT TOP 1 * FROM Products WHERE Price > 100
Product p = db.Products.Where(x => x.Price > 100).FirstOrDefault();
I'll add to this as I get further along the learning curve.

Update 08-01-2015

Get all Model Validation Errors via LINQ

This is very useful when debugging MVC Controller Methods.

var allErrors = ModelState.Values.SelectMany(v => v.Errors);

Tuesday, 22 July 2014

MVC 4 Display Correct Date Format in Edit Mode

This works:

@Html.TextBox("DateOrdered", string.Format("{0:d}", Model.DateOrdered))

Monday, 21 July 2014

MVC 4, EF5 Code First - Forcing a Decimal to x Number of Places

Stick this in your DB Context if you need to have EF recongise more than two decimal places.

  protected override void OnModelCreating(DbModelBuilder modelBuilder)  
     {  
       // We have a Decimal Field in Our SQL Table decimal(18, 4)  
       // By default only the first two digits after the decimal will be recognised if this property is set to type decimal  
       // we need to override this behaviour and here's how  
       modelBuilder.Entity<MyObjec t>().Property(f => f.totalWeightMg).HasPrecision(18,4);       
       base.OnModelCreating(modelBuilder);  
     }  

Wednesday, 9 July 2014

Return Yes, No from Boolean instead of True, False

I recently needed to output Boolean values into an iTextSharp PDF document having them represented as Yes or No rather than True or False

Cue MSDN. This article Boolean Structure explains how to do this.

Here's a quick example:

bool JobDone = true;
string strJobDone = JobDone ? "Yes" : "No"; // Returns "Yes"