Skip to content

Latest commit

 

History

History
251 lines (175 loc) · 7.27 KB

README.md

File metadata and controls

251 lines (175 loc) · 7.27 KB

Api Stack Net: the api accellerator

Build status nuget

ApiStackNet is a non-invasive api accellerator. Compared to other frameworks it doesn't exclude the possibility to create vanilla code. It's purpose is to:

  • avoid redundand code,
  • give a solid stack to all developers,
  • keep a clean architecture.

What is included out of the box:

  • three level architecture scaffold ( DAL, BLL, API)
  • swagger support
  • entity framework 6 support (database first or code first)
  • service pattern scaffold

With ApiStackNet you can scaffold a rest web service just writing the only part where you bring value. So, you just need to define data model and mapping it to the business model to get all working!

Getting Started

This guide drive the you through a simple CRUD to show basic framework features.

1. NEW PROJECT

create a new web api project called 'ApiStackNet.Demo'. Add the nuget package

Install-Package ApiStackNet

2. DAL

Inherith Base entity and configure EF mapping

    [Table("MYTABLE", Schema = "DBO")]
    public class MyTable : AuditableEntity<Int32>
    {        
        public string Field { get; set; }
    }

using auditable entity audit fields are included by design. To adapt field mapping just override fields.

    [Table("MYTABLE", Schema = "DBO")]
    public class MyTable : AuditableEntity<Int32>
    {   
		[Column("MY_CUSTOM_NAME")]
        public string Field { get; set; }

		public override DateTime CreatedOn { get; set; }
    }

3. BLL

There are two different class to implement:

  • BO (Business Object) is the class used to save data. This contains only relevant fields for saving
  • DTO (Data Transfert Object) is the class used to profide data, after business logic manipulation.
 public class MyTableDTO:AuditableEntity<Int32>
{
	public string EntityName { get; set; }
	//other computed or read-only fields here
}


    public class MyTableBO:BaseEntity<Int32>
    {
        public string EntityName { get; set; }
		//no computed fields here
		//no read only fields
		
    }

4. Mapping

We need two mapping class (entity -> DTO) (BO ->entity), used to fill DTO and save data.

public class MyTablegBO2EMapping:Profile
{
    public MyTablegBO2EMapping()
    {
        CreateMap<MyTableBO, MyTable>(); //NO reverse map.ReverseMap();
    }
}


public class MyTableE2DTOMapping : Profile
{
    public MyTableE2DTOMapping()
    {
        CreateMap<MyTable, MyTableDTO>();// No Reverse map ! .ReverseMap();
    }
}

Each mapping is implemented by a AutoMapper profile. This part can be also managed by using only one profile, but when classes grow may be harder to manage.

5. BLL

Business logic layer is impemented using Service Pattern. To get all basic CRUD functionality just implement a class:

 public class  MyTableDataService: IntDataService<MyTableDTO, MyTableBO,MyTable>
    {
        public  MyTableDataService(DbContext context, IMapper mapper) : base( context,  mapper)
        {
        
        }
    }

This configuration tell the relationship between entity, BO, DTO.

What class to inherit? There is one service for primary key type, naming convention is DataService, so you can have IntDataService, GuidDataService and so on.

6. Controller

Implementing a Controller extending DataController, this bring to you all CRUD functionalities.

    public class MyTableController : DataController<MyTableDataService, MyTableDTO, MyTableBO, MyTable, int>
    {
        public MyTableController(MyTableDataService service) : base(service)
        {
        }
    }

Each field can be searched from client. This work only for entity field. Related search (i.e. get all project of a given user) must be implemented by custom method or method extension.

{
  "PageNumber": 1,
  "PageSize": 10,
  "Filter": [
    {
      "Name": "FieldName",
      "Comparator": "Equal", 
      "Value": "true"
    }
  ]
}

7. AUTOMAPPER AND DBCONTEXT

download Automapper nuget package and create a class inside the new project called 'ApiStackNetDBContext', this is the primary class that is responsible for interacting with the database.

public class ApiStackNetDBContext:ApiStackDbContext
{
    public ApiStackNetDBContext() : base("ApiStackNetDBContext")
    {
        Database.SetInitializer<ApiStackNetDBContext>(null);
        this.Configuration.LazyLoadingEnabled = false;

        EnableVerboseSQLog();
    }

    //entities
    public DbSet<MyTable> MyTable { get; set; }
}

8. EXECUTE COMMANDS

you need to execute the following commands in the Package Manager Console in Visual Studio in this order:

-Enable-Migrations: Enables the migration in the project by creating a Configuration class. -Add-Migration: Creates a new migration class as per specified name with the Up() and Down() methods. -Update-Database: Executes the last migration file created by the Add-Migration command and applies changes to the database schema.

9. AUTOFAC

download Autofac nuget package and add a new class called 'AutofacConfig' and add this

using System.Web;
using System.Web.Compilation;
using System.Web.Http;

namespace ApiStackNet.Demo.App_Start
{
    public static class AutofacConfig
    {
        public static void RegisterContainer(HttpConfiguration config)
        {
            // Create your builder.
            var builder = new ContainerBuilder();

            builder.RegisterApiControllers(Assembly.GetExecutingAssembly());
            builder.RegisterControllers(Assembly.GetExecutingAssembly());
            builder.RegisterWebApiFilterProvider(config);
            
            builder.RegisterType<ApiStackNetDBContext>().As<DbContext>().InstancePerRequest();
            builder.RegisterType<MyTableDataService>().AsSelf().InstancePerRequest();
            builder.RegisterType<MessageService>().AsSelf().InstancePerRequest();
            builder.RegisterModule(new MapperInstaller());

            var container = builder.Build();

            Mapper.Initialize(x => x.ConstructServicesUsing(container.Resolve));

            config.DependencyResolver = new AutofacWebApiDependencyResolver(container);
        }
    }
}

and finally add this line at the beginning of the Application_Start() method of the Global.asax.cs class.

AutofacConfig.RegisterContainer(GlobalConfiguration.Configuration);

Build and Test

TEST Postman: The following tips refer to those test that need an "hand configuration" before to proceed:

  • GetById: set in the Request url bar the Id of the entity to get;
  • GetList: just send the request to run the test;
  • Edit: set in the Request url bar the Id of the entity to edit and check the corrispondence between the hand setted Id and the parameter "Id" in the body of the request;
  • Save: just send the request to run the test, using the json in the body;
  • Delete: set in the Request url bar the Id of the entity to delete;
  • DeleteByEntity: just send the request to run the test, using the json in the body.

Contribute

Any contribution is welcome, open an issue to start.

License

ApiStackNet is released under LGPL License.