I've been writing a script to publish services, and I am getting stuck on setting dynamic layers on a map service in Manager
I'd like to add this as a SOE I'm assuming to my script, but i'm not having any luck finding anything on how to do so.
We use scripts (python toolbox) to publish our map and geoprocessing services.
For most of our services, we use the default service setting. In order to set custom properties (like dynamic layers), we store the custom properties in a json config file. After the service is published, we fetch the service properties using the a call to the Admin API, edit the fetched json object using the stored file, then apply the edits by making another call to the Admin API.
BTW, our server is still at 10.1, but I don't see any reason why it wouldn't apply to 10.3.
Yeah, I'm not sure this would work for 10.3.1.
10.3 uses the .mxd --> .sddraft --> .sd --> Service publishing path logic, not .mxd --> .msd --> .sd --> Service like 10.1 does.
Thanks though
Hi Jacob,
Not sure if you saw this link with some samples. You can edit the .sddraft file using the xml.dom.minidom library. I was able to figure out how to enable dynamic layers, but have not been successful is setting the dynamic workspaces. Below is an example:
sddraft = r"E:\Temp\Python\Airports.sddraft"
doc = DOM.parse(sddraft)
# edit configuration properties
configProps = doc.getElementsByTagName('ConfigurationProperties')[0]
propArray = configProps.firstChild
propSets = propArray.childNodes
for propSet in propSets:
keyValues = propSet.childNodes
for keyValue in keyValues:
if keyValue.tagName == 'Key':
if keyValue.firstChild.data == "enableDynamicLayers":
keyValue.nextSibling.firstChild.data = "true"
if keyValue.firstChild.data == "dynamicDataWorkspaces":
#the below would fail
keyValue.nextSibling.firstChild.data = '[{"id":"MyFileGDBWorkspaceID","workspaceFactory":"FileGDB","workspaceConnection":"DATABASE=E:\\Temp\\Python\\Test.gdb"}]'
# output to a new sddraft
outXml = r"E:\Temp\Python\Airports_New.sddraft"
if os.path.exists(outXml): os.remove(outXml)
f = open(outXml, 'w')
doc.writexml( f )
f.close()Hopefully a python expert can way in to see how to update the 'dynamicDataWorkspaces' portion of the sddraft file.
Yeah that's the only part I've struggled with, I've setup a script that will disable KML, enable Feature Services and set Feature services to Query only in the .sddraft file prior to publishing. I just can't get the syntax to Enable Dynamic Workspaces.
.mxd --> .sddraft --> .sd --> Service
We're at 10.1 SP1, and the work flow is the same. .mxd --> .sddraft --> .sd --> Service.
Regardless, what I'm suggest occurs after the service is published. I will see if I can scrape together a sample script.
Here is a code sample demonstrating how to update service parameters after the service has been published.
#modify these to match your environment
serverHost = 'yourgisServer.com'
serverPort = 6080
serviceRelativePath = r'/services/serviceFolder/serviceName.MapServer'
username = 'yourAdminName'
password = 'yourAdminPW'
# properites you would like to apply to the service
newProperties = {
"minInstancesPerNode": 1,
"maxInstancesPerNode": 5,
"properties": {
"enableDynamicLayers": "true",
"dynamicDataWorkspaces": "[{\"id\":\"wsName\",\"workspaceFactory\":\"SDE\",\"lockVersion\":\"false\",\"workspaceConnection\":\"connectionProps\"}]"
}
}
import json
import urllib
import httplib
# ####################
# helper functions
# ####################
def getUpdatedDict(inDict,modDict):
def _getPathsAndValuesFromModDict(d,parentDict):
for k, v in d.iteritems():
if isinstance(v, dict):
_getPathsAndValuesFromModDict(v,parentDict)
else:
parentDict = v
_getPathsAndValuesFromModDict(modDict,inDict)
return inDict
def adminApiCall(strAdminApiUrl,objParams):
def _getToken():
tokenURL = "/arcgis/admin/generateToken"
params = urllib.urlencode({'username': username, 'password': password, 'client': 'requestip', 'f': 'json'})
headers = {"Content-type": "application/x-www-form-urlencoded", "Accept": "text/plain"}
httpConn = httplib.HTTPConnection(serverHost, serverPort)
httpConn.request("POST", tokenURL, params, headers)
response = httpConn.getresponse()
if (response.status != 200):
httpConn.close()
return
else:
data = response.read()
httpConn.close()
# Check that data returned is not an error object
if not _assertJsonSuccess(data):
return
# Extract the toke from it
token = json.loads(data)
return token['token']
def _assertJsonSuccess(data):
obj = json.loads(data)
if 'status' in obj and obj['status'] == "error":
#print "Error: JSON object returns an error. " + str(obj)
return False
else:
return True
token = _getToken()
if objParams:
objParams['token'] = token
objParams['f'] = 'json'
else:
objParams = {'token': token,'f': 'json'}
params = urllib.urlencode(objParams)
headers = {"Content-type": "application/x-www-form-urlencoded", "Accept": "text/plain"}
# Connect to URL and post parameters
httpConn = httplib.HTTPConnection(serverHost, serverPort)
httpConn.request("POST", r'/arcgis/admin'+ strAdminApiUrl, params, headers)
# Read response
response = httpConn.getresponse()
if (response.status != 200):
print response.status
httpConn.close()
output = "Error: Code "+str(response.status)+', '+str(response.reason)
else:
data = response.read()
# Check that data returned is not an error object
if not _assertJsonSuccess(data):
obj = json.loads(data)
output = obj['messages']
else:
dataObj = json.loads(data)
output = dataObj
httpConn.close()
try:
if output[0] == u'Authorization failed. This user account does not have the required privileges to access this application.':
return {'status':'error','msg':u'Authorization failed for user'}
except:
return output
# ####################
# end helper functions
# ####################
#fetch the parameters using the admin api
serviceInfo = adminApiCall(serviceRelativePath,{'f':'json'})
#update new parameters using the new properties
serviceInfo = getUpdatedDict(serviceInfo,newProperties)
#apply the updated params
adminApiCall('/services/test/testForDave.MapServer/edit',{'f':'json','service':json.dumps(serviceInfo)})Modifying properties after the service is published is, in my opinion, easier than doing so via the sd or sddraft.
I think this is going to the me in the right direction, I'm trying to achieve the same with a python script. I'm trying to make it easier to ensure my GIS users are publishing new services. I've figured out the syntax for everything else though.
Can't figure out how to edit my comments, I'm trying to do the same in a python script that you use on a new service.