How To: Add Impersonate User in Service Portal (us... - ServiceNow Community
mhedtke24
Tera Expert

Greetings,

I would like to share with you the ability to add "Impersonate User" to the Service Portal without having to clone the out-of-box Header Menu widget.

I would like to first thank Service Catalyst for initially creating a widget for Service Portal impersonation.  I have taken what they built in a widget for impersonating users and created another widget based on the out-of-box Knowledge Notification Entry widget to inject "Impersonate User" in the the menu drop down and then call the Impersonate User widget when clicked.

find_real_file.png

We will be creating two widgets to make this work.

Widget #1

Name: Impersonate User

ID: impersonate_user

HTML Template:

<div class="container-fluid" ng-init="getRecentImpersonation()">
  <div class="row">
    <div class="btn-group btn-group-sm btn-group-justified" role="group">
      <label class="btn btn-info btn-sm" ng-model="c.selectedAction" uib-btn-radio="'sp'">Service Portal</label>
      <label class="btn btn-primary btn-sm" ng-model="c.selectedAction" uib-btn-radio="'classic'">Classic UI</label>
    </div>
  </div>
  <div class="row top-buffer" ng-if="data.isImpersonating">
    <div class="btn-group btn-group-sm btn-group-justified" role="group">
      <label ng-click="impersonate(c.data.realUser)" class="btn btn-danger btn-lg btn-block">
        <i class="fa fa-stop m-r-sm"></i>
        STOP Impersonating</label>
    </div>
  </div>
  <div class="row top-buffer">
    <div class="record-picker">
      <sn-record-picker field="c.data.toImpersonate" table="'sys_user'" display-field="'name'" on-change="impersonate(c.data.toImpersonate.value)"
                        display-fields="'user_name'" search-fields="'name'" page-size="100" placeholder="Search for user" options="{cache: true, allowClear: false}"
                        default-query="'active=true^locked_out=false^web_service_access_only=false^ORweb_service_access_onlyISEMPTY'"
                        value-field="'sys_id'">
      </sn-record-picker>
    </div>
  </div>
  <div class="row top-buffer">
    <div class="panel panel-default">
      <div class="panel-heading">
        <h5 class="panel-title">
          ${Recent Impersonations}
        </h5>
      </div>
      <a ng-bind="rec.user_display_value" ng-repeat="rec in recentImpersonations" href="javascript:void(0)" class="list-group-item ng-binding ng-scope"
         ng-click="impersonate(rec.user_sys_id)"></a>
    </div>
  </div>
</div>

CSS - SCSS:

.top-buffer { 
  margin-top:20px;
}

Client Script:

function($scope, $http, $window) {
	/* widget controller */
	var c = this;
	c.data.toImpersonate = {};

	c.selectedAction = {};

	$scope.impersonate = function(userName){
		//If we don't have a user we can't impersonate
		if(!userName){
			return;
		}
		//Call to the impersonation api with username/sys_id
		$http.post("/api/now/ui/impersonate/" + userName, {}).success(function(){
			$scope.showError = false;
			if(c.selectedAction === "classic"){
				window.location = "/";
			}else if(c.selectedAction === "sp" || c.selectedAction === null){
				window.location.reload();
			}else{
				window.location.reload();
			}

		}).error(function(response){
			if(response.error){
				$scope.showError = true;
				$scope.error = response.error;
				console.warn("Impersonate Failed! with error:", response.error);
			}
		})                             
	};

	//call to the impersonation api Get recent impersonations
	$scope.getRecentImpersonation = function(){
		$http.get("/api/now/ui/impersonate/recent", {}).success(function(response){
			$scope.showError = false;
			$scope.recentImpersonations = response.result;
		}).error(function(response){
			if(response.error){
				$scope.showError = true;
				$scope.error = response.error;
				console.warn("Get Recent Impersonations Failed! with error:" + response.error);
			}
		})
	};
}

Server Script:

(function() {
	/* populate the 'data' object */
	/* e.g., data.table = $sp.getValue('table'); */
	data.isImpersonating =  new GlideImpersonate().isImpersonating();
	data.realUser =  gs.getImpersonatingUserName();
})();

 

Widget #2

Name: Impersonate User Entry

ID: impersonate_user_entry

Server Script:

(function() {
  /* populate the 'data' object */
  /* e.g., data.table = $sp.getValue('table'); */

	data.canImpersonate = new GlideImpersonate().canImpersonate(gs.getUserID());
	data.isImpersonating =  new GlideImpersonate().isImpersonating();
	data.realUser =  gs.getImpersonatingUserName();
	
	if (data.canImpersonate || data.isImpersonating) {
		data.show_menu_entry = true;
	}
	else {
		data.show_menu_entry = false;
	}
	
})();

Link Function:

function(scope) {
	
	var $uibModal = $injector.get('$uibModal');
	
	// Add entry to navbar
	if(scope.data.show_menu_entry && !$('#impersonation')[0]) {
		$('#sp-nav-bar ul li.hidden-xs.dropdown ul.dropdown-menu li:first').after('<li style="cursor: pointer;"><a role="link" id="impersonation">${Impersonate User}</a></li>');
		$('#sp-nav-bar ul li.visible-xs-block:first').after('<li class="visible-xs-block" style="cursor: pointer;"><a role="link" id="impersonation">${Impersonate User}</a></li>');
	}
	
	$("#impersonation").click(function() {
		$uibModal.open({
			templateUrl: 'impersonate-widget',
			scope: scope
		})
	});
}

Next you will want to open the Impersonate User Entry widget in the platform. In the related lists go to Angular ng-templates and click New.

ID: impersonate-widget

Template:

<div class="panel panel-default">
  <h3 class="padder-md">
      Impersonate User
    </h3>
  <div class="modal-header">
    <widget id="impersonate_user"></widget>
  </div>
</div>

 

Final Step

Open your service portal home page in the Service Portal Designer and add the Impersonate User Entry widget to the page. Now impersonate User will show in the menu drop down for users with the impersonate role.

Add icons to menu items

If you want to add icons like you see in the screen shot above, follow these steps:

  1. Open the Stock theme in the platform.
  2. Add the following to CSS variables:
#sp-nav-bar > ul:nth-child(2) > li.hidden-xs.dropdown.ng-scope.open > ul > li.ng-scope > a:before {
	content: "\f08b"; /* Sign Out */
    font-family: FontAwesome;
    padding-right: 5px;
}

#sp-nav-bar > ul:nth-child(2) > li.hidden-xs.dropdown.ng-scope.open > ul > li:nth-child(1) > a:before {
	content: "\f007"; /* Profile */
    font-family: FontAwesome;
    padding-right: 8px;
}

#impersonation:before {
	content: "\f06e"; /* Impersonate User */
    font-family: FontAwesome;
    padding-right: 6px;
    margin-left: -2px;
}
Comments
yashlimviya
Kilo Contributor

Exactly followed the steps mentioned, but unable to implement this in kingston instance. But working as expected in Jakarta instance. Please suggest the solution for the above problem. It is urgent. Thanks in advance.

mhedtke24
Tera Expert

I have tested this in Kingston and it is working on a developer instance for me.  This also works in London.  Would you be able to provide me with the issue you are experiencing on a Kingston instance?

yashlimviya
Kilo Contributor

Following the above steps did not make any change in the instance.

trevorlandau
Tera Explorer

Thanks for sharing this - worked as expected in Jakarta and London and is super handy. Do you have any suggestions on how I may be able to embed this in the (cloned) header widget itself? It'd be ideal to not have to add the widget to every portal page. Any help / advice is appreciated!

carlmarais
Kilo Explorer

This is fantastic! Great work.

Community Alums
Not applicable

I am pretty new to ServiceNow portal. Could someone please provide me with instructions on how to perform the last step of adding the Impersonate User Entry widget to the page using the Service Portal Designer?
TIA.

mhedtke24
Tera Expert

In the platform view in the left navigation go to Service Portal > Service Portal Configuration. This will open a new window, then click on Designer. You can then select the portal page you want and can then drag and drop the widget to the page.

Community Alums
Not applicable

But which page should I choose to drag the widget in so that it shows up under the name, profie, logout? Should it be the login page?

mhedtke24
Tera Expert

We put it on our home page, not the login page. Just know that you would have to go to your home page to use it. Otherwise you can add it to all pages should you choose to do so.

Community Alums
Not applicable

Thanks it worked. However, the icons dont show up now.

 

Added the script to stock as instructed.

Bharat Raj Tati
Tera Expert

I had followed the exact steps that were stated above. It's working good and my version is Madrid.

find_real_file.png

Vijayr119
Tera Contributor

I followed above step and worked well.. But I have noticed that, when we are in home page then only we can get the "Impersonate User" button below the profile. Incase if I opned any other page and try to see under profile and its not visible... Can this applied across entire service portal page?

Community Alums
Not applicable

This isn't true for me, atleast on my personal dev. instance. I can see the Impersonate option on every screen in the portal. See some screenshots below.

 

find_real_file.png

find_real_file.pngfind_real_file.png

Vijayr119
Tera Contributor

It will show but it will disappear if you refresh the screen.. 

mhedtke24
Tera Expert

You would need to add the widget to each page you want to be able to use it on.

Markusz
Tera Contributor

Hi,

For me stop impersonating is not working on London release.

mhedtke24
Tera Expert

Should work. I have it working in Madrid.

gtk
Mega Sage

thanks for the article and i followed the steps to create two widgets and after that from FINAL STEP is is a little bit confusing for me to follow .... can you please elaborate it.

thanks in advance

mhedtke24
Tera Expert

You just need to open your home page of your portal in page designer and add the widget for the impersonate entry to the page.

jody_klitzke
Kilo Expert

This works great! Thanks for sharing! 

 

Tony DiRienzo
Giga Guru

You can accomplish this by adding the widget to the footer of your theme.  Here you can see I have added the "Impersonate User Entry", and I have also added a feedback button widget so that both are visible on all pages, no matter what your entry point to the portal is:

find_real_file.png

You will need to update the Link Function in the Impersonate User Entry widget, though, because the header menu DOM element does not yet exist when the Link Function first runs. The code below will check that the header menu exists before trying to add the menu entry and set up the link action:

function(scope) {

	var $uibModal = $injector.get('$uibModal');

	var checkExist = setInterval(function() {
		if ($('#sp-nav-bar ul li.hidden-xs.dropdown ul.dropdown-menu li:first').length && $('#sp-nav-bar ul li.visible-xs-block:first').length) {
			// Add entry to navbar
			if(scope.data.show_menu_entry && !$('#impersonation')[0]) {
				$('#sp-nav-bar ul li.hidden-xs.dropdown ul.dropdown-menu li:first').after('<li style="cursor: pointer;"><a role="link" id="impersonation">${Impersonate User}</a></li>');
				$('#sp-nav-bar ul li.visible-xs-block:first').after('<li class="visible-xs-block" style="cursor: pointer;"><a role="link" id="impersonation">${Impersonate User}</a></li>');
			}

			$("#impersonation").click(function() {
				$uibModal.open({
					templateUrl: 'impersonate-widget',
					scope: scope
				})
			});
			clearInterval(checkExist);
		}
	}, 100); // check every 100ms

}
mhedtke24
Tera Expert

This is great. Thanks for sharing.

Dimitri Destin
Giga Guru

Hi all,

 

It seems that this customization doesn't work on Orlando.

Can you confirm?

If no, do you know what we have to modify?

 

Thanks!

Bharat Raj Tati
Tera Expert

it is working as expected in Orlando Release.

Brook3
Giga Expert

Thank you for sharing!! 

I get an error message. 

"GlideImpersonate is not allowed in scoped applications"

Am I doing something wrong?

Brian Lancaster
Tera Sage

It is not working for me in Orlando either.  All I get is two buttons that say Service Porta and Classic UI but click on them does nothing.

find_real_file.png

I have an idea to make at least end impersonate available in the portal since you cannot get out of it without logging out if the user you impersonated only has access to the portal.

Brian Lancaster
Tera Sage

Never mind figured it out. seems you have to put it in the top most section of your home page.

kshaw
Giga Guru

I have implemented this solution in our portal - running on Paris.

Everything seemed to be working but just discovered a problem.

The menu item visibility does not appear to be respecting roles that permit impersonation. So if a logged in general user selects that item in the user pulldown, then the system hangs and the impersonation popup does not appear (that seems to be respecting the lack of a role) and the portal hangs.

Seems like the menu item visability needs to bel driven by role permission.

How can that be implemented.

Kim Aly
Tera Contributor

Thank you for sharing!! 

I get an error message. 

"GlideImpersonate is not allowed in scoped applications"

Am I doing something wrong?

KevinConsultant
Tera Contributor

Has anyone experienced issue with Automated Testing after implementing this solution from @mhedtke24  ?

I'm having difficulty with both Cloud & Client ATF with the testing hanging around step #26 (confirming the Service Catalog Items has been submitted)

gcmalcato
Tera Contributor

@kshaw, i change function in server script on Impersonate User Entry widget:

 

(function() {
    /* populate the 'data' object */
    /* e.g., data.table = $sp.getValue('table'); */

    data.canImpersonate = new GlideImpersonate().canImpersonate(gs.getUserID());
    data.isImpersonating = new GlideImpersonate().isImpersonating();
    data.realUser = gs.getImpersonatingUserName();
    
    // Check if the user has the impersonator role
    var hasImpersonatorRole = gs.hasRole('impersonator');
    
    if ((data.canImpersonate && hasImpersonatorRole) || data.isImpersonating) {
        data.show_menu_entry = true;
    }
    else {
        data.show_menu_entry = false;
    }
    
})();

 

 
Version history
Last update:
‎09-07-2018 11:44 AM
Updated by: