This commit is contained in:
171
Controllers/RvlController.cs
Normal file
171
Controllers/RvlController.cs
Normal file
@@ -0,0 +1,171 @@
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using System.Threading.Tasks;
|
||||
using Microsoft.AspNetCore.Http;
|
||||
using Microsoft.AspNetCore.Mvc;
|
||||
using Microsoft.AspNetCore.Authorization;
|
||||
using Microsoft.EntityFrameworkCore;
|
||||
using rockfishCore.Models;
|
||||
using rockfishCore.Util;
|
||||
using System.ComponentModel.DataAnnotations;
|
||||
|
||||
namespace rockfishCore.Controllers
|
||||
{
|
||||
[Produces("text/plain")]
|
||||
[Route("rvl")]
|
||||
public class RvlController : Controller //RAVEN license controller
|
||||
{
|
||||
private readonly rockfishContext ct;
|
||||
|
||||
|
||||
public RvlController(rockfishContext context)
|
||||
{
|
||||
ct = context;
|
||||
}
|
||||
|
||||
public class dtoRavLicense
|
||||
{
|
||||
[Required]
|
||||
public string RegisteredTo { get; set; }
|
||||
[Required]
|
||||
public Guid DbId { get; set; }
|
||||
[Required]
|
||||
public long LicenseExpiration { get; set; }
|
||||
[Required]
|
||||
public long MaintenanceExpiration { get; set; }
|
||||
[Required]
|
||||
public List<dtoLicenseFeature> Features { get; set; }
|
||||
[Required]
|
||||
public long SiteId { get; set; }
|
||||
}
|
||||
public class dtoLicenseFeature
|
||||
{
|
||||
dtoLicenseFeature()
|
||||
{
|
||||
Count = 0;
|
||||
}
|
||||
//name of feature / product
|
||||
public string Feature { get; set; }
|
||||
//Optional count for items that require it
|
||||
public long Count { get; set; }
|
||||
}
|
||||
|
||||
|
||||
#if (DEBUG)
|
||||
private const string LICENSE_SERVER_URL = "http://localhost:3001/";
|
||||
public const string SUPPORT_EMAIL = "cardjohn@ayanova.com";
|
||||
#else
|
||||
private const string LICENSE_SERVER_URL = "https://rockfish.ayanova.com/";
|
||||
public const string SUPPORT_EMAIL="support@ayanova.com";
|
||||
#endif
|
||||
|
||||
[HttpPost]
|
||||
public async Task<IActionResult> Post([FromBody] dtoRavLicense l)
|
||||
{
|
||||
if (!ModelState.IsValid)
|
||||
{
|
||||
return BadRequest(ModelState);
|
||||
}
|
||||
|
||||
|
||||
//Get customer from site id
|
||||
var CustomerId = await ct.Site.AsNoTracking().Where(z => z.Id == l.SiteId).Select(z => z.CustomerId).FirstOrDefaultAsync();
|
||||
|
||||
if (CustomerId == 0)
|
||||
{
|
||||
return BadRequest($"Customer could not be found for site id {l.SiteId}");
|
||||
}
|
||||
|
||||
var Customer = await ct.Customer.AsNoTracking().Where(z => z.Id == CustomerId).FirstOrDefaultAsync();
|
||||
if (Customer == null)
|
||||
{
|
||||
return BadRequest($"Customer could not be found for customer id {CustomerId}");
|
||||
}
|
||||
|
||||
if (string.IsNullOrWhiteSpace(Customer.AdminEmail))
|
||||
{
|
||||
return BadRequest($"Customer does not have an Admin / License email address, no where to send key");
|
||||
}
|
||||
|
||||
if (!Customer.Active)
|
||||
{
|
||||
return BadRequest($"Customer {Customer.Name} is not Active, set to active and try again");
|
||||
}
|
||||
|
||||
if (Customer.DoNotContact)
|
||||
{
|
||||
return BadRequest($"Customer {Customer.Name} is set to DO NOT CONTACT, unable to process and send key");
|
||||
}
|
||||
|
||||
|
||||
var newLicense = new RavenKeyFactory.AyaNovaLicenseKey();
|
||||
|
||||
newLicense.RegisteredTo = l.RegisteredTo;
|
||||
newLicense.DbId = l.DbId;
|
||||
newLicense.LicenseExpiration = DateUtil.EpochToDate(l.LicenseExpiration);
|
||||
newLicense.MaintenanceExpiration = DateUtil.EpochToDate(l.MaintenanceExpiration);
|
||||
foreach (dtoLicenseFeature f in l.Features)
|
||||
{
|
||||
newLicense.Features.Add(new RavenKeyFactory.LicenseFeature() { Feature = f.Feature, Count = f.Count });
|
||||
}
|
||||
|
||||
//Everything seems to be in order generate the license, save it and send it
|
||||
|
||||
|
||||
|
||||
|
||||
var NewRequest = new TrialRequest();
|
||||
NewRequest.Email = r.Email;
|
||||
NewRequest.DbId = r.DbId;
|
||||
NewRequest.CompanyName = r.Company;
|
||||
NewRequest.ContactName = r.Contact;
|
||||
await ct.TrialRequest.AddAsync(NewRequest);
|
||||
await ct.SaveChangesAsync();
|
||||
NewRequest.EmailConfirmCode = NewRequest.Id.ToString() + FetchKeyCode.generate();
|
||||
await ct.SaveChangesAsync();
|
||||
var verifyUrl = LICENSE_SERVER_URL + $"rvr/verify/{NewRequest.EmailConfirmCode}";
|
||||
var body = $"Please verify your email address by clicking the link below or copy and pasting into a browser\r\n{verifyUrl}\r\n(If you did not request this you can ignore this message)";
|
||||
//send confirmation email
|
||||
RfMail.SendMessage("support@ayanova.com", NewRequest.Email, "AyaNova trial request email verification", body, false);
|
||||
//return Ok(new { Accepted = true });
|
||||
return Accepted();
|
||||
|
||||
}
|
||||
|
||||
[HttpGet("verify/{code}")]
|
||||
public async Task<IActionResult> GetVerify([FromRoute] string code)
|
||||
{
|
||||
//is there a valid trial request
|
||||
var req = await ct.TrialRequest.Where(z => z.EmailConfirmCode == code && z.DtFetched == null && z.Status == TrialRequest.TrialRequestStatus.NotSet).FirstOrDefaultAsync();
|
||||
if (req == null)
|
||||
{
|
||||
return new ContentResult
|
||||
{
|
||||
ContentType = "text/html",
|
||||
StatusCode = 200,
|
||||
Content = "<html><body><h4>Request not found</h4><p>There was no evaluation request associated with this verification code.</p></body></html>"
|
||||
};
|
||||
}
|
||||
|
||||
req.EmailValidated = true;
|
||||
await ct.SaveChangesAsync();
|
||||
|
||||
//notify *us*
|
||||
//http://localhost:3001/default.htm#!/trials/[id]
|
||||
var rfUrl = LICENSE_SERVER_URL + $"default.htm#!/trialEdit/{req.Id}";
|
||||
var body = $"Email address {req.Email} was just verified for {req.ContactName} at {req.CompanyName}.\r\nTrial key is ready to be processed now:\r\n{rfUrl}";
|
||||
//send confirmation email
|
||||
RfMail.SendMessage("support@ayanova.com", SUPPORT_EMAIL, "AyaNova trial request requiring action", body, false);
|
||||
|
||||
return new ContentResult
|
||||
{
|
||||
ContentType = "text/html",
|
||||
StatusCode = 200,
|
||||
Content = "<html><body><h4>Email validated!</h4><p>Your request is being processed and you will receive an email with instructions for starting your AyaNova evaluation.</p></body></html>"
|
||||
};
|
||||
}
|
||||
|
||||
|
||||
}//eoc
|
||||
}//eons
|
||||
@@ -36,7 +36,7 @@ namespace rockfishCore.Util
|
||||
#region license classes
|
||||
|
||||
//DTO object returned on license query
|
||||
internal class LicenseFeature
|
||||
public class LicenseFeature
|
||||
{
|
||||
//name of feature / product
|
||||
public string Feature { get; set; }
|
||||
@@ -53,7 +53,14 @@ namespace rockfishCore.Util
|
||||
{
|
||||
Features = new List<LicenseFeature>();
|
||||
RegisteredTo = UNLICENSED_TOKEN;
|
||||
Id = RegisteredTo;
|
||||
//Id = RegisteredTo;
|
||||
LicenseFormat = "8";
|
||||
|
||||
var vv = Math.Truncate((DateTime.UtcNow.Subtract(new DateTime(1970, 1, 1))).TotalSeconds);
|
||||
string sId = vv.ToString();
|
||||
if (sId.Contains(","))
|
||||
sId = sId.Split('.')[0];
|
||||
Id = $"00-{sId}";
|
||||
}
|
||||
|
||||
public bool IsEmpty
|
||||
@@ -169,7 +176,7 @@ namespace rockfishCore.Util
|
||||
// private static string SAMPLE_KEY = @"[KEY
|
||||
// {
|
||||
// ""Key"": {
|
||||
// ""LicenseFormat"": ""2018"",
|
||||
// ""LicenseFormat"": ""8"",
|
||||
// ""Id"": ""34-1516288681"", <----Customer id followed by key serial id
|
||||
// ""RegisteredTo"": ""Super TestCo"",
|
||||
// ""DBID"": ""df558559-7f8a-4c7b-955c-959ebcdf71f3"",
|
||||
@@ -210,13 +217,13 @@ namespace rockfishCore.Util
|
||||
|
||||
//Build a sample test key, sign it and return it
|
||||
AyaNovaLicenseKey k = new AyaNovaLicenseKey();
|
||||
k.LicenseFormat = "2018";
|
||||
k.LicenseFormat = "8";
|
||||
|
||||
var vv = Math.Truncate((DateTime.UtcNow.Subtract(new DateTime(1970, 1, 1))).TotalSeconds);
|
||||
string sId = vv.ToString();
|
||||
if (sId.Contains(","))
|
||||
sId = sId.Split('.')[0];
|
||||
k.Id = $"00-{sId}";
|
||||
// var vv = Math.Truncate((DateTime.UtcNow.Subtract(new DateTime(1970, 1, 1))).TotalSeconds);
|
||||
// string sId = vv.ToString();
|
||||
// if (sId.Contains(","))
|
||||
// sId = sId.Split('.')[0];
|
||||
// k.Id = $"00-{sId}";
|
||||
k.RegisteredTo = CompanyName;
|
||||
k.DbId = dbid;
|
||||
|
||||
@@ -232,66 +239,26 @@ namespace rockfishCore.Util
|
||||
k.Features.Add(new LicenseFeature() { Feature = ACCOUNTING_FEATURE_NAME, Count = 0 });
|
||||
k.Features.Add(new LicenseFeature() { Feature = SERVICE_TECHS_FEATURE_NAME, Count = 1000 });
|
||||
|
||||
return genKey(k);
|
||||
return GenerateRavenKey(k);
|
||||
}
|
||||
|
||||
|
||||
#region RAVEN test code for development
|
||||
|
||||
//Trial key magic number for development and testing, all other guids will be fully licensed
|
||||
private static Guid TEST_TRIAL_KEY_DBID = new Guid("{A6D18A8A-5613-4979-99DA-80D07641A2FE}");
|
||||
|
||||
|
||||
|
||||
public static string GetRavenTestKey(Guid dbid)
|
||||
{
|
||||
|
||||
//Build a sample test key, sign it and return it
|
||||
AyaNovaLicenseKey k = new AyaNovaLicenseKey();
|
||||
k.LicenseFormat = "2018";
|
||||
|
||||
var vv = Math.Truncate((DateTime.UtcNow.Subtract(new DateTime(1970, 1, 1))).TotalSeconds);
|
||||
string sId = vv.ToString();
|
||||
if (sId.Contains(","))
|
||||
sId = sId.Split('.')[0];
|
||||
k.Id = $"00-{sId}";
|
||||
k.RegisteredTo = "Best Besterson Inc.";
|
||||
k.DbId = dbid;
|
||||
|
||||
//add accounting and user features either way
|
||||
k.Features.Add(new LicenseFeature() { Feature = ACCOUNTING_FEATURE_NAME, Count = 0 });
|
||||
k.Features.Add(new LicenseFeature() { Feature = SERVICE_TECHS_FEATURE_NAME, Count = 1000 });
|
||||
|
||||
|
||||
//fake trial key or fake licensed key
|
||||
if (dbid == TEST_TRIAL_KEY_DBID)
|
||||
{
|
||||
k.MaintenanceExpiration = k.LicenseExpiration = DateTime.UtcNow.AddMonths(1);
|
||||
k.Features.Add(new LicenseFeature() { Feature = TRIAL_FEATURE_NAME, Count = 0 });
|
||||
}
|
||||
else
|
||||
{
|
||||
k.MaintenanceExpiration = DateTime.UtcNow.AddYears(1);
|
||||
k.LicenseExpiration = DateUtil.EmptyDateValue;//1/1/5555 as per spec
|
||||
}
|
||||
|
||||
return genKey(k);
|
||||
}
|
||||
|
||||
#endregion
|
||||
|
||||
|
||||
/// <summary>
|
||||
/// New RAVEN key generator, so far just for testing purposes
|
||||
/// RAVEN key generator
|
||||
/// </summary>
|
||||
/// <returns></returns>
|
||||
private static string genKey(AyaNovaLicenseKey k)
|
||||
internal static string GenerateRavenKey(AyaNovaLicenseKey k)
|
||||
{
|
||||
|
||||
|
||||
if (string.IsNullOrWhiteSpace(k.RegisteredTo))
|
||||
throw new ArgumentException("RegisteredTo is required", "RegisteredTo");
|
||||
|
||||
if (k.DbId == Guid.Empty)
|
||||
throw new ArgumentException("DBId is required", "RegisteredTo");
|
||||
|
||||
try
|
||||
{
|
||||
@@ -438,28 +405,53 @@ oArP0E2Vbow3JMxq/oeXmHbrLMLQfYyXwFmciLFigOtkd45bfHdrbA==
|
||||
|
||||
|
||||
|
||||
// private static void AddLicensePlugin(Newtonsoft.Json.JsonWriter w, string pluginName, DateTime pluginExpires)
|
||||
// {
|
||||
|
||||
// //this dictionary is used by the additional message code to
|
||||
// //make the human readable portion of the license
|
||||
// _plugins.Add(pluginName, pluginExpires);
|
||||
|
||||
// //this is adding it to the actual key
|
||||
// w.WriteStartObject();
|
||||
// w.WritePropertyName("Item");
|
||||
// w.WriteValue(pluginName);
|
||||
|
||||
// w.WritePropertyName("SubscriptionExpires");
|
||||
// w.WriteValue(pluginExpires);
|
||||
|
||||
// w.WriteEndObject();
|
||||
// //----------------
|
||||
// }
|
||||
|
||||
|
||||
|
||||
//eoc
|
||||
}
|
||||
//eons
|
||||
}
|
||||
}
|
||||
|
||||
// #region RAVEN test code for development
|
||||
|
||||
// //Trial key magic number for development and testing, all other guids will be fully licensed
|
||||
// private static Guid TEST_TRIAL_KEY_DBID = new Guid("{A6D18A8A-5613-4979-99DA-80D07641A2FE}");
|
||||
|
||||
|
||||
|
||||
// public static string GetRavenTestKey(Guid dbid)
|
||||
// {
|
||||
|
||||
// //Build a sample test key, sign it and return it
|
||||
// AyaNovaLicenseKey k = new AyaNovaLicenseKey();
|
||||
// k.LicenseFormat = "8";
|
||||
|
||||
// var vv = Math.Truncate((DateTime.UtcNow.Subtract(new DateTime(1970, 1, 1))).TotalSeconds);
|
||||
// string sId = vv.ToString();
|
||||
// if (sId.Contains(","))
|
||||
// sId = sId.Split('.')[0];
|
||||
// k.Id = $"00-{sId}";
|
||||
// k.RegisteredTo = "Best Besterson Inc.";
|
||||
// k.DbId = dbid;
|
||||
|
||||
// //add accounting and user features either way
|
||||
// k.Features.Add(new LicenseFeature() { Feature = ACCOUNTING_FEATURE_NAME, Count = 0 });
|
||||
// k.Features.Add(new LicenseFeature() { Feature = SERVICE_TECHS_FEATURE_NAME, Count = 1000 });
|
||||
|
||||
|
||||
// //fake trial key or fake licensed key
|
||||
// if (dbid == TEST_TRIAL_KEY_DBID)
|
||||
// {
|
||||
// k.MaintenanceExpiration = k.LicenseExpiration = DateTime.UtcNow.AddMonths(1);
|
||||
// k.Features.Add(new LicenseFeature() { Feature = TRIAL_FEATURE_NAME, Count = 0 });
|
||||
// }
|
||||
// else
|
||||
// {
|
||||
// k.MaintenanceExpiration = DateTime.UtcNow.AddYears(1);
|
||||
// k.LicenseExpiration = DateUtil.EmptyDateValue;//1/1/5555 as per spec
|
||||
// }
|
||||
|
||||
// return genKey(k);
|
||||
// }
|
||||
|
||||
// #endregion
|
||||
@@ -273,6 +273,33 @@ app.api = (function () {
|
||||
});
|
||||
};
|
||||
|
||||
|
||||
///////////////////////////////////////////////////////////
|
||||
//CreateRavLicense
|
||||
//Route app.post('/api/license/create', function (req, res) {
|
||||
//
|
||||
createRavLicense = function (objData, callback) {
|
||||
$.ajax({
|
||||
method: "post",
|
||||
dataType: "text",
|
||||
url: app.shell.stateMap.apiUrl + "license/generate",
|
||||
headers: getAuthHeaderObject(),
|
||||
contentType: "application/json; charset=utf-8",
|
||||
data: JSON.stringify(objData),
|
||||
success: function (data, textStatus) {
|
||||
callback(data);
|
||||
},
|
||||
error: function (jqXHR, textStatus, errorThrown) {
|
||||
callback({
|
||||
error: 1,
|
||||
msg: textStatus + "\n" + errorThrown,
|
||||
error_detail: {}
|
||||
});
|
||||
}
|
||||
});
|
||||
};
|
||||
|
||||
|
||||
///////////////////////////////////////////////////////////
|
||||
//GetLicenseRequests
|
||||
//Fetch license requests
|
||||
|
||||
156
wwwroot/js/app.ravlicense.js
Normal file
156
wwwroot/js/app.ravlicense.js
Normal file
@@ -0,0 +1,156 @@
|
||||
/*
|
||||
* app.license.js
|
||||
* License key generator
|
||||
*/
|
||||
|
||||
/*jslint browser : true, continue : true,
|
||||
devel : true, indent : 2, maxerr : 50,
|
||||
newcap : true, nomen : true, plusplus : true,
|
||||
regexp : true, sloppy : true, vars : false,
|
||||
white : true
|
||||
*/
|
||||
|
||||
/*global $, app */
|
||||
|
||||
app.ravlicense = (function () {
|
||||
'use strict';
|
||||
//---------------- BEGIN MODULE SCOPE VARIABLES --------------
|
||||
var
|
||||
|
||||
stateMap = {},
|
||||
configModule, initModule, onGenerate, onSelectAllAddOns;
|
||||
//----------------- END MODULE SCOPE VARIABLES ---------------
|
||||
|
||||
//------------------- BEGIN UTILITY METHODS ------------------
|
||||
//-------------------- END UTILITY METHODS -------------------
|
||||
|
||||
|
||||
//------------------- BEGIN EVENT HANDLERS -------------------
|
||||
onGenerate = function (event) {
|
||||
|
||||
event.preventDefault();
|
||||
$.gevent.publish('app-clear-error');
|
||||
//get form data
|
||||
var formData = $("form").serializeArray({
|
||||
checkboxesAsBools: true
|
||||
});
|
||||
|
||||
var submitData = app.utilB.objectifyFormDataArray(formData);
|
||||
|
||||
app.api.createRavLicense(submitData, function (res) {
|
||||
if (res.error) {
|
||||
$.gevent.publish('app-show-error', res.msg);
|
||||
} else {
|
||||
$('#key').val(res);
|
||||
return false;
|
||||
}
|
||||
});
|
||||
|
||||
return false; //prevent default
|
||||
};
|
||||
|
||||
|
||||
onSelectAllAddOns = function (event) {
|
||||
event.preventDefault();
|
||||
$('#wbi').prop('checked', true);
|
||||
$('#mbi').prop('checked', true);
|
||||
$('#ri').prop('checked', true);
|
||||
$('#qbi').prop('checked', true);
|
||||
$('#qboi').prop('checked', true);
|
||||
$('#pti').prop('checked', true);
|
||||
$('#quickNotification').prop('checked', true);
|
||||
$('#exportToXls').prop('checked', true);
|
||||
$('#outlookSchedule').prop('checked', true);
|
||||
$('#oli').prop('checked', true);
|
||||
$('#importExportCSVDuplicate').prop('checked', true);
|
||||
|
||||
|
||||
|
||||
return false; //prevent default
|
||||
};
|
||||
|
||||
// onTemplates = function(event) {
|
||||
// event.preventDefault();
|
||||
// alert("STUB: templates");
|
||||
|
||||
|
||||
// return false; //prevent default
|
||||
// };
|
||||
|
||||
|
||||
//-------------------- END EVENT HANDLERS --------------------
|
||||
|
||||
//------------------- BEGIN PUBLIC METHODS -------------------
|
||||
//CONFIGMODULE
|
||||
//
|
||||
configModule = function (context) {
|
||||
stateMap.context = context.context;
|
||||
if (stateMap.context.params.id) {
|
||||
stateMap.id = stateMap.context.params.id;
|
||||
}
|
||||
};
|
||||
|
||||
//INITMODULE
|
||||
//
|
||||
initModule = function ($container) {
|
||||
if (typeof $container === 'undefined') {
|
||||
$container = $('#app-shell-main-content');
|
||||
}
|
||||
$container.html(Handlebars.templates['app.license']({}));
|
||||
|
||||
|
||||
//case 3233 customer list
|
||||
//Fill customer list combo
|
||||
var customerList = {};
|
||||
|
||||
|
||||
//get customers
|
||||
app.api.get('customer/list', function (res) {
|
||||
if (res.error) {
|
||||
$.gevent.publish('app-show-error', res.msg);
|
||||
} else {
|
||||
|
||||
var html = '<option value="0"><TRIAL></option>';
|
||||
|
||||
for (var i = 0, len = res.length; i < len; ++i) {
|
||||
html += ('<option value="' + res[i]['id'] + '">' + res[i]['name'] + '</option>');
|
||||
customerList[res[i]['id']] = res[i]['name'];
|
||||
}
|
||||
$('#customerId').append(html);
|
||||
}
|
||||
});
|
||||
|
||||
|
||||
//Context menu
|
||||
app.nav.contextClear();
|
||||
|
||||
////app.nav.setContextTitle("License");
|
||||
|
||||
//make context menu
|
||||
|
||||
|
||||
//Context menu
|
||||
app.nav.contextClear();
|
||||
app.nav.contextAddButton('btn-generate', 'Make', 'key', onGenerate);
|
||||
app.nav.contextAddButton('btn-select-all-addons', 'All', 'check-all', onSelectAllAddOns);
|
||||
// app.nav.contextAddLink("licenseRequests/", "Requests", "voice");
|
||||
app.nav.contextAddLink("licenseTemplates/", "", "layers");
|
||||
//case 3233
|
||||
app.nav.contextAddLink("licenses/", "List", "");
|
||||
|
||||
|
||||
//set all date inputs to today plus one year
|
||||
var oneYearFromNow = moment().add(1, 'years').toISOString().substring(0, 10);
|
||||
var oneMonthFromNow = moment().add(1, 'months').toISOString().substring(0, 10);
|
||||
$('input[type="date"]').val(oneYearFromNow);
|
||||
$('#lockoutDate').val(oneMonthFromNow);
|
||||
|
||||
};
|
||||
|
||||
// return public methods
|
||||
return {
|
||||
configModule: configModule,
|
||||
initModule: initModule
|
||||
};
|
||||
//------------------- END PUBLIC METHODS ---------------------
|
||||
}());
|
||||
183
wwwroot/js/templates/app.ravlicense.handlebars
Normal file
183
wwwroot/js/templates/app.ravlicense.handlebars
Normal file
@@ -0,0 +1,183 @@
|
||||
<div>
|
||||
<form id="frm" method="post" action="index.html">
|
||||
<div class="row">
|
||||
|
||||
|
||||
<div class="col-sm-6">
|
||||
<div class="form-check">
|
||||
<label class="form-check-label" for="keyWillLockout">
|
||||
<input class="form-check-input" type="checkbox" name="keyWillLockout" id="keyWillLockout"> Expiring test key</label>
|
||||
</div>
|
||||
<input class="form-control" type="date" id="lockoutDate" name="lockoutDate" value="">
|
||||
</div>
|
||||
|
||||
<div class="col-sm-6">
|
||||
<div class="form-check">
|
||||
<label class="form-check-label" for="isLite">
|
||||
<input class="form-check-input" type="checkbox" name="isLite" id="isLite"> AyaNova LITE</label>
|
||||
</div>
|
||||
</div>
|
||||
<div class="col-sm-6">
|
||||
<div class="form-group">
|
||||
<label for="licenseType">License type</label>
|
||||
<select class="form-control" name="licenseType">
|
||||
<option value="new">New</option>
|
||||
<option value="addon">Renewal / Add-on</option>
|
||||
<option value="licensedTrial">Licensed trial</option>
|
||||
<option value="webRequestedTrial">Web requested trial</option>
|
||||
</select>
|
||||
</div>
|
||||
</div>
|
||||
<div class="col-sm-6">
|
||||
<div class="form-group">
|
||||
<label for="registeredTo">Registered to</label>
|
||||
<input class="form-control" type="text" id="registeredTo" name="registeredTo" value="">
|
||||
</div>
|
||||
</div>
|
||||
|
||||
|
||||
<div class="col-sm-6">
|
||||
<div class="form-group">
|
||||
<label for="customerId">Customer</label>
|
||||
<select class="form-control" id="customerId" name="customerId" />
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="col-sm-6">
|
||||
<div class="form-group">
|
||||
<label for="emailAddress">Email address</label>
|
||||
<input class="form-control" type="text" id="emailAddress" name="emailAddress" value="">
|
||||
</div>
|
||||
</div>
|
||||
|
||||
|
||||
<div class="col-sm-6">
|
||||
<div class="form-group">
|
||||
<label for="users">Users</label>
|
||||
<select class="form-control" name="users">
|
||||
<option value="1">1</option>
|
||||
<option value="5">5</option>
|
||||
<option value="10">10</option>
|
||||
<option value="15">15</option>
|
||||
<option value="20">20</option>
|
||||
<option value="50">50</option>
|
||||
<option value="999">999</option>
|
||||
</select>
|
||||
</div>
|
||||
</div>
|
||||
<div class="col-sm-6">
|
||||
|
||||
<div class="form-group">
|
||||
<label for="supportExpiresDate">Support expires</label>
|
||||
<input class="form-control" type="date" id="supportExpiresDate" name="supportExpiresDate" value="">
|
||||
</div>
|
||||
</div>
|
||||
<div class="col-sm-6">
|
||||
<div class="form-check">
|
||||
<label class="form-check-label" for="wbi">
|
||||
<input class="form-check-input" type="checkbox" name="wbi" id="wbi"> WBI
|
||||
</label>
|
||||
<input class="form-control" type="date" id="wbiSupportExpiresDate" name="wbiSupportExpiresDate" value="">
|
||||
</div>
|
||||
</div>
|
||||
<div class="col-sm-6">
|
||||
<div class="form-check">
|
||||
<label class="form-check-label" for="mbi">
|
||||
<input class="form-check-input" type="checkbox" name="mbi" id="mbi"> MBI
|
||||
</label>
|
||||
<input class="form-control" type="date" id="mbiSupportExpiresDate" name="mbiSupportExpiresDate" value="">
|
||||
</div>
|
||||
</div>
|
||||
<div class="col-sm-6">
|
||||
<div class="form-check">
|
||||
<label class="form-check-label" for="ri">
|
||||
<input class="form-check-input" type="checkbox" name="ri" id="ri"> RI
|
||||
</label>
|
||||
<input class="form-control" type="date" id="riSupportExpiresDate" name="riSupportExpiresDate" value="">
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="col-sm-6">
|
||||
<div class="form-check">
|
||||
<label class="form-check-label" for="qbi">
|
||||
<input class="form-check-input" type="checkbox" name="qbi" id="qbi"> QBI
|
||||
</label>
|
||||
<input class="form-control" type="date" id="qbiSupportExpiresDate" name="qbiSupportExpiresDate" value="">
|
||||
</div>
|
||||
</div>
|
||||
|
||||
|
||||
|
||||
<div class="col-sm-6">
|
||||
<div class="form-check">
|
||||
<label class="form-check-label" for="qboi">
|
||||
<input class="form-check-input" type="checkbox" name="qboi" id="qboi"> QBOI
|
||||
</label>
|
||||
<input class="form-control" type="date" id="qboiSupportExpiresDate" name="qboiSupportExpiresDate" value="">
|
||||
</div>
|
||||
</div>
|
||||
<div class="col-sm-6">
|
||||
<div class="form-check">
|
||||
<label class="form-check-label" for="pti">
|
||||
<input class="form-check-input" type="checkbox" name="pti" id="pti"> PTI
|
||||
</label>
|
||||
<input class="form-control" type="date" id="ptiSupportExpiresDate" name="ptiSupportExpiresDate" value="">
|
||||
</div>
|
||||
</div>
|
||||
<div class="col-sm-6">
|
||||
<div class="form-check">
|
||||
<label class="form-check-label" for="quickNotification">
|
||||
<input class="form-check-input" type="checkbox" name="quickNotification" id="quickNotification"> Quick notification</label>
|
||||
<input class="form-control" type="date" id="quickNotificationSupportExpiresDate" name="quickNotificationSupportExpiresDate"
|
||||
value="">
|
||||
</div>
|
||||
</div>
|
||||
<div class="col-sm-6">
|
||||
<div class="form-check">
|
||||
<label class="form-check-label" for="exportToXls">
|
||||
<input class="form-check-input" type="checkbox" name="exportToXls" id="exportToXls"> Export to XLS</label>
|
||||
<input class="form-control" type="date" id="exportToXlsSupportExpiresDate" name="exportToXlsSupportExpiresDate" value="">
|
||||
</div>
|
||||
</div>
|
||||
<div class="col-sm-6">
|
||||
<div class="form-check">
|
||||
<label class="form-check-label" for="outlookSchedule">
|
||||
<input class="form-check-input" type="checkbox" name="outlookSchedule" id="outlookSchedule"> Outlook Schedule</label>
|
||||
<input class="form-control" type="date" id="outlookScheduleSupportExpiresDate" name="outlookScheduleSupportExpiresDate" value="">
|
||||
</div>
|
||||
</div>
|
||||
<div class="col-sm-6">
|
||||
<div class="form-check">
|
||||
<label class="form-check-label" for="oli">
|
||||
<input class="form-check-input" type="checkbox" name="oli" id="oli"> OLI
|
||||
</label>
|
||||
<input class="form-control" type="date" id="oliSupportExpiresDate" name="oliSupportExpiresDate" value="">
|
||||
</div>
|
||||
</div>
|
||||
<div class="col-sm-6">
|
||||
<div class="form-check">
|
||||
<label class="form-check-label" for="importExportCSVDuplicate">
|
||||
<input class="form-check-input" type="checkbox" name="importExportCSVDuplicate" id="importExportCSVDuplicate"> Import / export CSV duplicate</label>
|
||||
<input class="form-control" type="date" id="importExportCSVDuplicateSupportExpiresDate" name="importExportCSVDuplicateSupportExpiresDate"
|
||||
value="">
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="col-sm-12">
|
||||
<div class="form-group">
|
||||
<label for="key">Key</label>
|
||||
<textarea class="form-control" id="key" name="key" rows="10" />
|
||||
</div>
|
||||
</div>
|
||||
|
||||
</div>
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
<div class="app-frm-buttons">
|
||||
|
||||
</div>
|
||||
</form>
|
||||
</div>
|
||||
Reference in New Issue
Block a user