using Disco.Models.Repository; using Disco.Models.Services.Devices; using Disco.Models.Services.Documents; using Disco.Models.Services.Jobs.JobLists; using Disco.Models.UI.Device; using Disco.Services; using Disco.Services.Authorization; using Disco.Services.Devices.DeviceFlags; using Disco.Services.Exporting; using Disco.Services.Interop.ActiveDirectory; using Disco.Services.Plugins.Features.DetailsProvider; using Disco.Services.Plugins.Features.UIExtension; using Disco.Services.Users; using Disco.Services.Web; using Disco.Web.Models.Shared; using System; using System.Collections.Generic; using System.Data.Entity; using System.Linq; using System.Web.Mvc; namespace Disco.Web.Controllers { public partial class DeviceController : AuthorizedDatabaseController { #region Index public virtual ActionResult Index() { Models.Device.IndexModel m = new Models.Device.IndexModel(); // UI Extensions UIExtensions.ExecuteExtensions(this.ControllerContext, m); return View(); } #endregion #region Add Offline [DiscoAuthorize(Claims.Device.Actions.EnrolDevices)] public virtual ActionResult AddOffline() { var m = new Models.Device.AddOfflineModel() { DefaultDeviceProfileId = Database.DiscoConfiguration.DeviceProfiles.DefaultAddDeviceOfflineDeviceProfileId }; if (Authorization.Has(Claims.Device.Properties.DeviceBatch)) m.DeviceBatches = Database.DeviceBatches.ToList(); if (Authorization.Has(Claims.Device.Properties.DeviceProfile)) { m.DeviceProfiles = Database.DeviceProfiles.ToList(); if (m.DefaultDeviceProfileId == 0) { m.DeviceProfiles.Insert(0, new DeviceProfile() { Id = 0, Name = "Select a Device Profile" }); } } // UI Extensions UIExtensions.ExecuteExtensions(this.ControllerContext, m); return View(m); } [DiscoAuthorize(Claims.Device.Actions.EnrolDevices), HttpPost] public virtual ActionResult AddOffline(Models.Device.AddOfflineModel m) { // Trim Serial Number & Error Check m.Device.SerialNumber = m.Device.SerialNumber.Trim(); if (string.IsNullOrEmpty(m.Device.SerialNumber)) { ModelState.AddModelError("Device.SerialNumber", "The Serial Number is Required"); } else if (m.Device.SerialNumber.Contains("/") || m.Device.SerialNumber.Contains(@"\")) { ModelState.AddModelError("Device.SerialNumber", @"The Serial Number cannot contain '/' or '\' characters"); } else { // Ensure Existing Device Doesn't Exist if (!string.IsNullOrEmpty(m.Device.SerialNumber) && Database.Devices.Count(d => d.SerialNumber == m.Device.SerialNumber) > 0) ModelState.AddModelError("Device.SerialNumber", "A Device what this Serial Number already exists"); } if (string.IsNullOrWhiteSpace(m.Device.DeviceDomainId)) m.Device.DeviceDomainId = null; if (m.Device.DeviceDomainId != null) { try { if (ActiveDirectory.IsValidDomainAccountId(m.Device.DeviceDomainId, out var accountUsername, out var accountDomain)) { m.Device.DeviceDomainId = $@"{accountDomain.NetBiosName}\{accountUsername}"; } else { ModelState.AddModelError("Device.DeviceDomainId", @"The computer name must be in the format 'DOMAIN\Username'"); } } catch (ArgumentException ex) { ModelState.AddModelError("Device.DeviceDomainId", ex.Message); } } if (ModelState.IsValid) { var d = m.Device.AddOffline(Database); Database.SaveChanges(); return RedirectToAction(MVC.Device.Show(d.SerialNumber)); } return AddOffline(); } #endregion #region Export [HttpGet] [DiscoAuthorize(Claims.Device.Actions.Export)] public virtual ActionResult Export(Guid? exportId, DeviceExportTypes? exportType, int? exportTypeTargetId) { var m = new Models.Device.ExportModel() { Options = Database.DiscoConfiguration.Devices.LastExportOptions, DeviceBatches = new KeyValuePair[] { new KeyValuePair(0, "") }.Concat(Database.DeviceBatches.OrderBy(db => db.Name).Select(db => new { Key = db.Id, Value = db.Name }).ToList().Select(i => new KeyValuePair(i.Key, i.Value))), DeviceModels = Database.DeviceModels.OrderBy(dm => dm.Description).Select(dm => new { Key = dm.Id, Value = dm.Description }).ToList().Select(i => new KeyValuePair(i.Key, i.Value)), DeviceProfiles = Database.DeviceProfiles.OrderBy(dp => dp.Name).Select(dp => new { Key = dp.Id, Value = dp.Name }).ToList().Select(i => new KeyValuePair(i.Key, i.Value)) }; m.Fields = ExportFieldsModel.Create(m.Options, DeviceExportOptions.DefaultOptions()); m.Fields.AddCustomUserDetails(o => o.UserDetailCustom, m.Fields.FieldGroups.FindIndex(g => g.Name == "Assigned User") + 1); if (ExportTask.TryFromCache(exportId, out var context)) { m.ExportId = context.Id; m.ExportResult = context.Result; } if (exportType.HasValue && exportTypeTargetId.HasValue) { m.Options.ExportType = exportType.Value; m.Options.ExportTypeTargetId = exportTypeTargetId.Value; } // UI Extensions UIExtensions.ExecuteExtensions(ControllerContext, m); return View(m); } #endregion #region Import [DiscoAuthorize(Claims.Device.Actions.Import), HttpGet] public virtual ActionResult Import(string Id) { var m = new Models.Device.ImportModel() { DeviceModels = Database.DeviceModels.ToList(), DeviceProfiles = Database.DeviceProfiles.ToList(), DeviceBatches = Database.DeviceBatches.ToList() }; if (!string.IsNullOrWhiteSpace(Id)) m.CompletedImportSessionContext = Areas.API.Controllers.DeviceController.Import_RetrieveContext(Id, Remove: true); // UI Extensions UIExtensions.ExecuteExtensions(this.ControllerContext, m); return View(m); } [DiscoAuthorize(Claims.Device.Actions.Import), HttpGet] public virtual ActionResult ImportHeaders(string Id) { if (string.IsNullOrWhiteSpace(Id)) throw new ArgumentNullException("Id"); var context = Areas.API.Controllers.DeviceController.Import_RetrieveContext(Id); if (context == null) throw new ArgumentException("The Import Session Id is invalid or the session timed out (60 minutes), try importing again", "Id"); var m = new Models.Device.ImportHeadersModel() { Context = context }; // UI Extensions UIExtensions.ExecuteExtensions(this.ControllerContext, m); return View(m); } [DiscoAuthorize(Claims.Device.Actions.Import), HttpGet] public virtual ActionResult ImportReview(string Id) { if (string.IsNullOrWhiteSpace(Id)) throw new ArgumentNullException("Id"); var context = Areas.API.Controllers.DeviceController.Import_RetrieveContext(Id); if (context == null) throw new ArgumentException("The Import Session Id is invalid or the session timed out (60 minutes), try importing again", "Id"); var m = new Models.Device.ImportReviewModel() { Context = context, StatisticErrorRecords = context.Records.Count(r => r.HasError), StatisticNewRecords = context.Records.Count(r => r.RecordAction == System.Data.EntityState.Added), StatisticModifiedRecords = context.Records.Count(r => r.RecordAction == System.Data.EntityState.Modified), StatisticUnmodifiedRecords = context.Records.Count(r => r.RecordAction == System.Data.EntityState.Unchanged) }; // UI Extensions UIExtensions.ExecuteExtensions(this.ControllerContext, m); return View(m); } #endregion #region Show [DiscoAuthorize(Claims.Device.Show)] public virtual ActionResult Show(string id) { var m = new Models.Device.ShowModel(); Database.Configuration.LazyLoadingEnabled = true; m.Device = Database.Devices .Include(d => d.DeviceModel) .Include(d => d.DeviceProfile) .Include(d => d.DeviceBatch) .Include(d => d.DeviceDetails) .Include(d => d.DeviceUserAssignments.Select(a => a.AssignedUser.UserFlagAssignments)) .Include(d => d.AssignedUser.UserFlagAssignments) .Include(d => d.AssignedUser.UserDetails) .Include(d => d.DeviceCertificates) .Include(d => d.DeviceAttachments.Select(a => a.TechUser)) .Include(d => d.DeviceAttachments.Select(a => a.DocumentTemplate)) .Include(d => d.DeviceFlagAssignments.Select(a => a.AddedUser)) .Include(d => d.DeviceFlagAssignments.Select(a => a.RemovedUser)) .FirstOrDefault(d => d.SerialNumber == id); if (m.Device == null) throw new ArgumentException(string.Format("Unknown Device: [{0}]", id), "id"); // No Necessary - Yet... //if (!string.IsNullOrWhiteSpace(m.Device.ComputerName)) //{ // var adMachineAccount = BI.Interop.ActiveDirectory.ActiveDirectory.GetMachineAccount(m.Device.ComputerName); // if (adMachineAccount != null) // { // m.OrganisationUnit = adMachineAccount.ParentDistinguishedName; // } //} if (Authorization.Has(Claims.Device.Properties.DeviceProfile)) { var profiles = Database.DeviceProfiles.Select(dp => new { Profile = dp, DecommissionedCount = dp.Devices.Count(d => d.DecommissionedDate.HasValue), DeviceCount = dp.Devices.Count(), }).ToList(); m.DeviceProfiles = profiles.Select(p => p.Profile).OrderBy(p => p.Name).ToList(); m.DecommissionedDeviceProfileIds = new HashSet(profiles.Where(p => p.DeviceCount > 0 && p.DecommissionedCount == p.DeviceCount).Select(p => p.Profile.Id)); } if (Authorization.Has(Claims.Device.Properties.DeviceBatch)) { var batches = Database.DeviceBatches.Select(db => new { Batch = db, DecommissionedCount = db.Devices.Count(d => d.DecommissionedDate.HasValue), DeviceCount = db.Devices.Count(), }).ToList(); m.DeviceBatches = batches.Select(b => b.Batch).OrderBy(b => b.Name).ToList(); m.DecommissionedDeviceBatchIds = new HashSet(batches.Where(b => b.DeviceCount > 0 && b.DecommissionedCount == b.DeviceCount).Select(b => b.Batch.Id)); } if (Authorization.Has(Claims.Device.ShowJobs)) { m.Jobs = new JobTableModel() { ShowStatus = true, ShowDevice = false, IsSmallTable = false, HideClosedJobs = true, EnablePaging = false }; m.Jobs.Fill(Database, Disco.Services.Searching.Search.BuildJobTableModel(Database).Where(j => j.DeviceSerialNumber == m.Device.SerialNumber).OrderByDescending(j => j.Id), true); } if (Authorization.Has(Claims.Device.ShowCertificates)) m.Certificates = Database.DeviceCertificates.Where(c => c.DeviceSerialNumber == m.Device.SerialNumber).ToList(); if (Authorization.Has(Claims.Device.Actions.GenerateDocuments)) { m.DocumentTemplates = m.Device.AvailableDocumentTemplates(Database, UserService.CurrentUser, DateTime.Now); m.DocumentTemplatePackages = m.Device.AvailableDocumentTemplatePackages(Database, UserService.CurrentUser); } m.DeviceProfileDefaultOrganisationAddress = m.Device.DeviceProfile.DefaultOrganisationAddressDetails(Database); if (m.Device.DeviceProfile.CertificateProviders != null) { m.DeviceProfileCertificateProviders = m.Device.DeviceProfile.GetCertificateProviders().ToList(); } if (m.Device.DeviceProfile.WirelessProfileProviders != null) { m.DeviceProfileWirelessProfileProviders = m.Device.DeviceProfile.GetWirelessProfileProviders().ToList(); } if (Authorization.Has(Claims.Device.ShowFlagAssignments)) { var usedFlags = m.Device.DeviceFlagAssignments .Where(a => !a.RemovedDate.HasValue) .Select(a => a.DeviceFlagId) .Distinct().ToList(); m.AvailableDeviceFlags = DeviceFlagService.GetDeviceFlags().Where(f => !usedFlags.Contains(f.Id)).ToList(); } if (Authorization.Has(Claims.User.ShowDetails)) { // Populate Custom Details m.PopulateDetails(Database); } // UI Extensions UIExtensions.ExecuteExtensions(this.ControllerContext, m); return View(m); } #endregion } }