Monday, December 28, 2015

Quick and Dirty OCR on OS X for Free

Wait, Stop, Don't spend that $80+ on some fancy OCR software just yet.  There are some free open source tools out there and with a bit of work you can have very functional workflow using them and OS X. This guide will help you set up an applet you can drop images onto to OCR them to plain text.  This will not create PDF's, formatted, or indent documents, but rather just one large text block that is structured the same as the source text with line breaks.  The following tools are used, but instructions are provided below so you don't need to download anything from the sites.
  • Homebrew The missing package manager for OS X
  • Tesseract Open Source OCR Engine
  • Automator

Friday, December 4, 2015

Monday, October 19, 2015

Thursday, October 8, 2015

Friday, October 2, 2015

Problems with Java Based OS X Applications Hanging when Entering Text

So I found another problem with my new upgrade to OS X 10.11.  That is that the Java web application I was using, Adobe Ad Hoc Analysis in this case, would hard lock every time i tried to enter text in a text box.  The only solution was to force quit the application.  After much testing I found the culprit was the RescueTime helper applet.  Basically this applet gets Accessibility access so that it can inspect windows and record time used but is glitching on the java windows and causing the lock up.

There could be other programs causing the same issues even if you don't use RescueTime but are having the same problem.  To troubleshoot do the following.

  • Close out of the java program that is locking up.
  • Open System Preference > Security & Privacy > Privacy > Accessibility.
  • Note with items are check and then uncheck them all.
  • Restart your java program and attempt to reproduce the issue.
  • If it is working correctly then quit, re-enable one item, and try again until you find the culprit.
  • If you can't get around the error even with all Accessibility options off that you must have some other issue.
Anyway if you find any items fix the problem you can just leave them off though they might hinder there applications functionality.

OSX 10.11 Stuck On Grey Screen After Login

After upgrading to El Capitan yesterday I ran into an issue where my computer would boot up going through the loading bar and display the login screen if  enabled.  Upon logging in, or automatic login, the system would hang at a grey screen.  After some testing with the network I figured out that it was not actually locking up and would respond to traffic.

To determine if your issues is due to the same problem wither SSH into the computer once booted if remote login was enabled or boot to Recovery Mode  and launch the Terminal.  Then run the following to see recently logged errors.

tail /var/log/system.log | grep 'Segmentation fault'

Or if using the Recovery Mode terminal the following thought the drive name may be different.

tail /Volumes/Macintosh\ HD/var/log/system.log | grep 'Segmentation fault'

If either of these commands produce any results that you have found your problem.  The culprit is Unsigned Kernel Extensions or .kext modules.  In the good old days developers could just install these as needed to add code level kernel functionality but no for improved security OS X requires that all Kernel Extensions be signed by the developer.  This is overall a good thing because these are privileged and could easily lead to malicious behavior by untrusted sources.

Anyway the problem where is that the upgrade didn't disable the unsigned Kernel Extensions it found and instead just doest't load them.  This is fine and normally it would just mean that whatever applications depend on them would jus not work until updated but for whatever reason there are a few special cases that cause the login process to hang.  I ended up removing all the Virtual Box and Soundflower modules and then was able to boot without issue.  I have not tried to Re-install or upgrade wither yet but atlas my computer boots.

To remove these again go to the Terminal and do the following.  You will nee to authorize them with your administrative password.

sudo rm -Rf /System/Library/Extensions/Waco*
sudo rm -Rf /Library/Extensions/VBox*
sudo rm -Rf /Library/Application\ Support/VirtualBox/*
sudo rm -Rf /System/Library/Extensions/Soundflower*
sudo rm -Rf /var/folders/*

Again note the additional prefix if using the Recovery Mode ant that thedick name may be different.

sudo rm -Rf /Volumes/Macintosh\ HD/System/Library/Extensions/Waco*
sudo rm -Rf /Volumes/Macintosh\ HD/Library/Extensions/VBox*
sudo rm -Rf /Volumes/Macintosh\ HD/Library/Application\ Support/VirtualBox/*
sudo rm -Rf /Volumes/Macintosh\ HD/System/Library/Extensions/Soundflower*
sudo rm -Rf /Volumes/Macintosh\ HD/var/folders/*

Finally just reboot and enjoy.

Tuesday, September 29, 2015

Raspberry Pi B Crashing when Running on SD Card

I have been having surtax crashes with my Raspberry Pi B for a few months now and it is a very irritating intermittent issue.  The unit basically locks up and even on resetting the power it won't always turn back on, but rather just sit there with no lights and no display and I end of having to cycle the power 2-5 times to get it to start again.

Anyway, I finally figured out there it appears to be an issues with the SD card.  At first I thought that it was just an off brand card and the Raspberry Pi just didn't like it.  But after some testing and hands on trials I figured out that it appeared to be an actual problem with the electrical connection and even though the card seamed seated properly to poor connection would cause the card to be or become unreadable.  This was easy to verify by lightly pressing on the SD card and then turning on the Raspberry PI.  It turned right on and worked perfectly and as soon as I let go it died again.  I can't see any physical defects and the card looks to be seated and pressing firmly into the contacts but I guess it's not sufficient.




I found a simple solution, a simple light pressure chip clip/close pin on the SD-card solved the problem perfectly.  I added a bit of cloth above the SD card and ensured that the clasp was pressing just above the SD Card's contact points and placed it over the end of the Raspberry PI and it works like a charm.

Wednesday, September 16, 2015

Friday, July 31, 2015

Drag and Drop in Umbraco 7

This code will let you add drag and drop logic to Umbraco v7 using the new AngularJS interface.  It doesn't require and backend code updates but can't be created as a package because there is no way to update the umbTree directive as needed.  There are only two files that need to be changed on the client.  Note that updates to umbraco will overwrite these changes.  Also note that due to lack of support in the underlying jQuery-sortable plugin for touch events this will not work on tablet devices.

First a set of changes for the umbTree and umbTreeItem directives in umbraco/Js/umbraco.directives.js.
--- umbraco.directives-a449b728.js
+++ (clipboard)
@@ -2388,7 +2388,7 @@
* @name umbraco.directives.directive:umbTree
* @restrict E
**/
-function umbTreeDirective($compile, $log, $q, $rootScope, treeService, notificationsService, $timeout, userService) {
+function umbTreeDirective($compile, $log, $q, $rootScope, treeService, notificationsService, $timeout, userService, $injector) {
return {
restrict: 'E',
@@ -2420,8 +2420,8 @@
'<a href="#/{{section}}" ng-click="select(tree.root, $event)" class="root-link">{{tree.name}}</a></h5>' +
'<a class="umb-options" ng-hide="tree.root.isContainer || !tree.root.menuUrl" ng-click="options(tree.root, $event)" ng-swipe-right="options(tree.root, $event)"><i></i><i></i><i></i></a>' +
'</div>';
- template += '<ul>' +
- '<umb-tree-item ng-repeat="child in tree.root.children" eventhandler="eventhandler" node="child" current-node="currentNode" tree="this" section="{{section}}" ng-animate="animation()"></umb-tree-item>' +
+ template += '<ul ui-sortable="sortableOptions" class="item" ng-model="tree.root.children">' +
+ '<umb-tree-item ng-repeat="child in tree.root.children" eventhandler="eventhandler" node="child" current-node="currentNode" tree="this" section="{{section}}" ng-animate="animation()" sortable-options="sortableOptions"></umb-tree-item>' +
'</ul>' +
'</li>' +
'</ul>';
@@ -2644,6 +2644,7 @@
.then(function(data) {
//set the data once we have it
scope.tree = data;
+ scope.node = scope.tree.root;
enableDeleteAnimations();
@@ -2789,7 +2790,160 @@
lastSection = newVal;
}
});
-
+ scope.sortableOptions = {
+ connectWith: ".item",
+ cursor: "move",
+ items: '>li',
+ axis: 'y',
+ tolerance: 'pointer',
+ containment: '.umb-tree .root>ul',
+ disabled: !scope.section.match("content|media") || scope.isdialog === "true",
+ update: function (e, ui) {
+ var node = ui.item.scope() ? ui.item.scope().node : scope.dragCurrentNode;
+
+ scope.newParentNode = $(e.target.parentElement).scope().node;
+
+ // Ignore if this is just a sort order change.
+ if (node.parentId == scope.newParentNode.id) {
+ scope.dragCurrentNode = node;
+ return;
+ }
+ //console.log("id:" + node.id + "; parent:" + node.parentId + "; newParent:" + $(e.target.parentElement).scope().node.id);
+
+ //Now we need to check if this is for media or content because that will depend on the resources we use
+ var contentResource, contentTypeResource;
+ if (scope.section === "media") {
+ contentResource = $injector.get('mediaResource');
+ contentTypeResource = $injector.get('mediaTypeResource');
+ }
+ else if (scope.section === "content") {
+ contentResource = $injector.get('contentResource');
+ contentTypeResource = $injector.get('contentTypeResource');
+ } else {
+ return;
+ }
+
+ if (scope.newParentNode.id == -20 || scope.newParentNode.id == -21)
+ // Delete the node
+ contentResource.deleteById(node.id)
+ .then(function () {
+ scope.loadChildren(scope.newParentNode, true);
+ });
+ else
+ // Move node, this will automaticaly validate the move.
+ contentResource.move({ parentId: scope.newParentNode.id, id: node.id })
+ .then(function () {
+ // Sync client side ui changes. Tis will relaod the relevent part of the tree. It could probubly jsut update the new with the new values instead and simplify the back and forth a bit.
+ scope.dragMoved = true;
+
+ // Reload collapsed destination.
+ if ($(e.target.parentElement).children("ul.item").is(".collapsed")) {
+ scope.loadChildren(scope.newParentNode, true);
+ }
+ }, function (err) {
+ // Reload source and destination on a invalide move.
+ scope.loadChildren(scope.newParentNode, true);
+ scope.loadChildren(ui.item.scope().node.parent(), true);
+ })
+ },
+ start: function (e, ui) {
+ // Store the original sort order.
+ scope.newParentNode = ui.item.scope().node.parent();
+ scope.originalSort = _.map(
+ _.filter(
+ scope.newParentNode.children,
+ function (item) {
+ return parseInt(item.id) > -1;
+ }),
+ function (item) {
+ return item.id;
+ });
+
+ // Tempararily enable collapsed nodes as valid targets for putting items inside them without expanding them.
+ ui.item.parents(".umb-tree").addClass("ui-dragging")
+ },
+ stop: function (e, ui) {
+ // Remove extra elements for child dropping.
+ ui.item.parents(".umb-tree").removeClass("ui-dragging")
+
+ // Update sort order for all children of the parent node.
+ var sortOrder = _.map(
+ _.filter(
+ scope.newParentNode.children
+ , function (item) {
+ return parseInt(item.id) > -1;
+ })
+ , function (item) {
+ return item.id;
+ });
+
+ // Don't do anything if there are no changes.
+ if (sortOrder.join() === scope.originalSort.join()) {
+ return;
+ }
+
+ // Don't do anything if this is moved to a collapsed node.
+ if (sortOrder.join() == "") {
+ return;
+ }
+
+
+ //Now we need to check if this is for media or content because that will depend on the resources we use
+ /*var contentResource, contentTypeResource;
+ if (scope.section === "media") {
+ contentResource = $injector.get('mediaResource');
+ contentTypeResource = $injector.get('mediaTypeResource');
+ }
+ else if (scope.section === "content") {
+ contentResource = $injector.get('contentResource');
+ contentTypeResource = $injector.get('contentTypeResource');
+ } else {
+ return;
+ }*/
+
+ // Post new sort order
+ $timeout(function () {
+ //console.log("id:" + scope.newParentNode.id + "; sort:" + sortOrder.join());
+
+ // Use custom UpdateSortOrder service privided by native Sort tool as the contentResource.sort() service doesn't appear to save changes properly.
+ $.ajax({
+ type: "POST",
+ url: "/umbraco/WebServices/NodeSorter.asmx/UpdateSortOrder?app=" + scope.section,
+ data: '{ "ParentId": ' + parseInt(scope.newParentNode.id) + ', "SortOrder": "' + sortOrder.join() + '"}',
+ contentType: "application/json; charset=utf-8",
+ dataType: "json",
+ success: function (msg) {
+ scope.complete = true;
+
+ // reload parent to get a clean node.
+ if (scope.dragMoved) {
+ scope.loadChildren(scope.newParentNode, true);
+ //$(e.target.parentElement).scope().loadChildren(scope.newParentNode, true);
+ scope.dragMoved = false;
+ }
+ }
+ });
+
+
+ /* // This is a more direct way to save sort changes but it appears to be non-functinal in 7.2.8
+ contentResource.sort({ parentId: scope.newParentNode.id, sortedIds: sortOrder })
+ .then(function () {
+ console.log("Sort Done");
+ scope.complete = true;
+
+ // reload parent to get a clean node.
+ if (scope.dragMoved) {
+ scope.loadChildren(scope.newParentNode, true);
+ //$(e.target.parentElement).scope().loadChildren(scope.newParentNode, true);
+ scope.dragMoved = false;
+ }
+ });*/
+ // This delay is to allow the move to finish before we resort. We can get the sort index untill the stop()
+ // so we can't attach it to the move promise, thus the short delay.
+ }, 250, false);
+ }
+ };
+
setupExternalEvents();
loadTree();
};
@@ -2829,13 +2983,14 @@
eventhandler: '=',
currentNode: '=',
node: '=',
- tree: '='
+ tree: '=',
+ sortableOptions: '='
},
//TODO: Remove more of the binding from this template and move the DOM manipulation to be manually done in the link function,
// this will greatly improve performance since there's potentially a lot of nodes being rendered = a LOT of watches!
- template: '<li ng-class="{\'current\': (node == currentNode)}" on-right-click="altSelect(node, $event)">' +
+ template: '<li ng-class="{\'current\': (node == currentNode)}" on-right-click="altSelect(node, $event)" >' +
'<div ng-class="getNodeCssClass(node)" ng-swipe-right="options(node, $event)" >' +
//NOTE: This ins element is used to display the search icon if the node is a container/listview and the tree is currently in dialog
//'<ins ng-if="tree.enablelistviewsearch && node.metaData.isContainer" class="umb-tree-node-search icon-search" ng-click="searchNode(node, $event)" alt="searchAltText"></ins>' +
@@ -3021,7 +3176,7 @@
setupNodeDom(scope.node, scope.tree);
- var template = '<ul ng-class="{collapsed: !node.expanded}"><umb-tree-item ng-repeat="child in node.children" eventhandler="eventhandler" tree="tree" current-node="currentNode" node="child" section="{{section}}" ng-animate="animation()"></umb-tree-item></ul>';
+ var template = '<ul ui-sortable="sortableOptions" class="item" ng-model="node.children" ng-class="{collapsed: !node.expanded}" ><umb-tree-item ng-repeat="child in node.children" eventhandler="eventhandler" tree="tree" current-node="currentNode" node="child" section="{{section}}" ng-animate="animation()" sortable-options="sortableOptions"></umb-tree-item></ul>';
var newElement = angular.element(template);
$compile(newElement)(scope);
element.append(newElement);

 This patch was generated on 7.2.8 so may not work on other versions automatically.  You can manually apply all the listed changes on any 7.* version and they should work as expected with no changes needed.

Secondly, to allow for dragging items to collapsed nodes or nodes they don't already have children, the following needs to be added to the end of the line in umbraco/assets/css/umbraco.css.
.umb-tree.ui-dragging ul.collapsed{display:block;height:5px;margin-bottom:-5px;overflow:hidden;}
view raw umbraco.css hosted with ❤ by GitHub

Friday, May 29, 2015

For Each to loop over HttpFileCollection Files

Recently I was working with file uploads in .NET and found that theres no clean way to loop over each uploaded file.  Here is a extension method for easily looping through the list and it allows filtering by field name and applying Regex to the file name.

Samples:

Request.Files.Each(
(file, name) => model.otherLogos = model.otherLogos.Concat(new string[] {file.FileName}).ToArray(),
"otherLogos"
);
Request.Files.Each(
(file, name) => file.SaveAs(String.Format("{0}/{2}", rootPath, file.FileName)),
"otherLogos",
new Regex(".*(?<!exe|php|aspx|asp|cs|vp|dll|scr)$") // Prevent uploading specified exstentions.
);

Source Code

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Text.RegularExpressions;
using System.Web;
namespace
{
public static class HttpFileCollectionExtensions
{
/// <summary>
/// Loop through each file and raise a callback function.
/// </summary>
/// <param name="self">File Collection to loop through.</param>
/// <param name="callback">Callback function to call and pass file name and form field name.</param>
/// <param name="fieldFilter">If provided only include files posted to this form field.</param>
/// <param name="fileNameFilter">If provided only include files matching this expression.</param>
public static void Each(this HttpFileCollection self, Action<HttpPostedFile, string> callback, string fieldFilter = "", Regex fileNameFilter = null)
{
for (int i = 0; i < self.Count; i++)
{
if (String.IsNullOrEmpty(fieldFilter) || self.Keys[i] == fieldFilter)
{
if (fileNameFilter == null || fileNameFilter.IsMatch(self[i].FileName))
{
callback.Invoke(self[i], self.Keys[i]);
}
}
}
}
}
}

Friday, May 22, 2015

Convert .htaccess deny rules to web.config

Recently I was looking for a easy way to convert a large list of spammer and hacker IP Addresses from Apache .htaccess allow/deny rules into a format I could use on our IIS Server.  I found that there is really no good way to do this but after some work and a bit of coding here is a solution.

This solution will allow you to take a large list of access rules and apply them to an IIS website.  Additionally this is done in a file so there is no need to manually add records.

Collect your List of Rules

I used the following as a general blacklist of malicious and spammer client's to block.  There are other sources or you may have your own list.  Compile all the lists into one file and don't worry about duplicates, the script later on will resolve those.

Convert the Rules to Web.config Equivalents

I have created a simple javascript tool to convert the above lists into web.config rules.  To do this go here and enter all your rules into the first text box.  Make sure you supply all rules at once so that they can be de-duped.  If the list does end up having duplicates then IIS will not start up properly.

Ensure you have the Features Installed

Next log onto your server and make sure that you have the Web Server > Security  > IP and Domain Restrictions role installed.  You may need to add it as it's not selected by default.

Apply your new Rules

Finally open up the web.config files for each of your main sites and add the generated text to the system.webServer/security/ipSecurity section. Save and reload the website to ensure it's working.  

Example Web.config File

<?xml version="1.0" encoding="utf-8"?>
<configuration>
...
<system.webServer>
<security>
<ipSecurity>
<!-- Cambodia (KH) -->
<add ipAddress="114.134.184.0" subnetMask="255.255.248.0" />
<!-- Chinese (CN) IP addresses follow: -->
<add ipAddress="1.80.0.0" subnetMask="255.248.0.0" />
<add ipAddress="1.92.6.4" />
</ipSecurity>
</security>
</system.webServer>
</configuration>
view raw web.config.xml hosted with ❤ by GitHub

Tuesday, May 5, 2015

Manually replace a SSL Certificates in IIS 7

Recently I was trying to update SSL Certificated on our Windows Server on IIS7 and when trying to complete the Certificate Request I was getting a cryptic error.

CertEnroll::Cx509Enrollment::p_InstallResponse: ASN1 bad tag value met.

In short I wasn't able to find much help on this but after much fiddling I found an alternate procedure to import SSL Certificates in PKCS#7 or  X.509 formats.

Instead of trying to import the certificate directly in IIS add it to the certificate story manual by doing the following.  Then refresh and it will show up in IIS as expected.

  1. Save the certificate and make sure it is accessible on the server.
  2. On the server open up MMC.
    1. Start > Run > MMC
  3. Add Certificates Snap-in
    1. File > Add/Remove Snap-in
    2. Select Certificates > Add > My User Account > Add
    3. Select Certificates > Add > Computer Account > Next > Finish
    4. Click Ok
  4. Check in both Current User and Local Computer for old certificates to remove
    1. Look under Personal > Certificates
    2. Remove the old certificate you are trying to replace by selecting it and pressing delete.
  5. Import the new certificate by either method
    1. Double Click it in the file browser and choose Install Certificate
    2. Under Local Computer > Personal > Right Click and select All Task > Import.
  6. Move the new certificate to the proper store
    1. Expand either Current User > Personal > Certificates or Current User > Other People > Certificates to find the new certificate.
    2. Verify the expression date so ensure you have found the new certificate.
    3. Drag the certificate into the Local Computer > Personal > Certificates store.
  7. Look up the Certificate Thumbprint
    1. Now that there certificate is in the proper location Double Click on it to view the details.
    2. Click on Details > Edit Properties and set the Common Name to the proper domain name.
    3. Ok to go back and then scroll down to find the Thumbprint and copy this to your clipboard.
  8. Import the Private Key
    1. Open an elevated Command Prompt
      1. Start > Type command > Right Click > Run As Administrator
    2. Enter the following command replacing sample Thumbprint with the value you looked up above.  Make sure to keep the quotations.
    3. certutil –repairstore my “00 01 02 03 04 05 06 07 08 09 0a 0b 0c 0d 0e 0f”
    4. Now go back to the Certificate loaded into the Personal store and refresh and you will see a little key icon next to it.  Without this it won't be able to sign pages in IIS.
  9. Now if you go back to IIS you will see your certificate
    1. If you don't see the new certificate then it either wasn't moved the the proper location or doesn't have the private key required to function.

Monday, March 23, 2015

Prevent Google Chromes on OS X from navigating when scrolling

I have been having this very irritating issue when using the Apple Magic Mouse and Google Chrome.  No matter what I tried when I would use the vertical finer scrolling gesture to scroll up and down the page Chrome would go back and forward in the page history when scrolling to the top or bottom of the page.  This was vary frustrating and is apparently a but in the Chrome touch implementation as it happened in no other browser.  After much headaches, I could just turn off two finger navigation but I use it in other browsers where it works correctly, I finally found a solution.

The Solution:

Open up Terminal and Enter the following.
defaults write com.google.Chrome AppleEnableMouseSwipeNavigateWithScrolls -bool false
<enter>

Tada!

Thursday, January 29, 2015

jQuery .ensure() Function to Wait for a Condition Before Executing Code

Overview

This is a very nice little plugin I wrote for jQuery to handle situations that don't have callbacks.  You can do this manually in code by adding a timeout but what if the function takes to long, or what make users with a faster device wait for slower devices by over estimating the timeout.

Please note that most libraries have their own callbacks and jQuery had a $().done() mechanism that will do this for common functions and libraries.  This solution is for those odd ball code bits that don't or that you don't have control over.

It takes a few pieces of information:
  • selector
    • The root jQuery selector you are binding to, it will become the this in the isValid call.
    • If your waiting for a new item to be created do $(document).ensure(selector).done().
  • isValid (default $(this).exists())
    • This is the function that is run each iteration.
    • It should be optimized and return a bool result.  Ie don't walk the DOM every time, select the proper parent node into a variable or pass as the selector.
  • delay (default 500 milliseconds)
    • Miliseconds between each iteration.
  • tries (default 0)
    • Maximum number of times to try, 0 for unlimited.

Examples:

$(document).ensure(function() {
return $("p.test").exists();
}).done(function() {
console.log("p.test exisits.");
});
$("body").ensure(function() {
return $("a", this).exists();
}).done(function(self) {
$("a", self).toggle();
});
$(selector).ensure({
delay: 500,
tries: 0,
isValid: function() {
return $(this).is(".loaded");
}
}).done(function(self) {
console.log("done loading resource");
}).fail(function() {
console.log("failed to load resource");
});
view raw ensure-demo.js hosted with ❤ by GitHub

Source Code:

/* Add $().exists() helper method.
Usage:
$(selector).exists()
*/
jQuery.fn.exists = function(){return this.length>0;}
/* Add $().ensure() helper method that ensures conditions pass before executing.
This will re-check the conditions untill they are meet or the limit is reached.
Usage:
$(document).ensure(function() {
return $("p.test").exists();
}).done(function() {
console.log("p.test exisits.");
});
$("body").ensure(function() {
return $("a", this).exists();
}).done(function(self) {
$("a", self).toggle();
});
$(selector).ensure({
delay: 500,
tries: 0,
isValid: function() {
return $(this).is(".loaded");
}
}).done(function(self) {
console.log("done loading resource");
}).fail(function() {
console.log("failed to load resource");
});
*/
jQuery.fn.ensure = function(options) {
if(typeof(options) == "function")
options = { isValid: options };
if(typeof(options) == "string") {
var selector = options;
options = { isValid: function() {
return $(selector, this).exists();
}};
}
options = $.extend({
delay: 500,
tries: 0,
isValid: function() {
return $(this).exists();
}
}, options);
var self = this;
var ensurePromise = new $.Deferred();
var loopCount = 0;
var looper = setInterval(function () {
loopCount++;
if (options.isValid.apply(self)) {
clearInterval(looper);
ensurePromise.resolve(self);
} else if(options.tries > 0 && loopCount >= options.tries) {
clearInterval(looper);
ensurePromise.reject(self);
}
}, options.delay);
return ensurePromise;
};