API(Application Programming Interface,应用程序接口): 是一些预先定义的接口(如函数、HTTP接口),或指软件系统不同组成部分衔接的约定。 用来提供应用程序与开发人员基于某软件或硬件得以访问的一组例程,而又无需访问源码,或理解内部工作机制的细节。
K8s也提供API接口,提供这个接口的是管理节点的apiserver组件,apiserver服务负责提供HTTP API,以便用户、其他组件相互通信。
有两种方式可以操作K8s中的资源:
K8s支持三种客户端身份认证:
实际上官方使用kubeconfig 文件,也是用的里面的token,这个方式不建议,因为带了根证书。自己开发跳过这块逻辑,所以可以用这个文件登录了。
安装Kubernetes客户端库:
pip install kubernetes -i https://pypi.tuna.tsinghua.edu.cn/simple
HTTPS证书认证(kubeconfig):
# 默认使用: cat ./kube/config
from kubernetes import client, config
import os
kubeconfig = os.path.join(os.getcwd(),"kubeconfig") # 获取当前目录并拼接文件
config.load_kube_config(kubeconfig) # 指定kubeconfig配置文件(/root/.kube/config)
apps_api = client.AppsV1Api() # 资源接口类实例化
#通过 kubectl api-resources | grep deploy 判断是哪个资源接口
for dp in apps_api.list_deployment_for_all_namespaces().items:
print(dp) # 打印Deployment对象详细信息
HTTP Token认证(ServiceAccount):
from kubernetes import client
import os
configuration = client.Configuration()
configuration.host = "https://192.168.31.61:6443" # APISERVER地址
ca_file = os.path.join(os.getcwd(),"ca.crt") # K8s集群CA证书(/etc/kubernetes/pki/ca.crt)
configuration.ssl_ca_cert= ca_file
configuration.verify_ssl = True # 启用证书验证,启用的话 要指定ca_file 的配置
token = "eyJhbGciOiJSUzI1NiIsImtpZCI6ImdlQlFUM3..." # 指定Token字符串,下面方式获取
configuration.api_key = {"authorization": "Bearer " + token} # Bearer后的空格不能少
client.Configuration.set_default(configuration)
apps_api = client.AppsV1Api()
for dp in apps_api.list_deployment_for_all_namespaces().items:
print(dp)
获取Token字符串:创建service account并绑定默认cluster-admin管理员集群角色:
# 创建用户
kubectl create serviceaccount dashboard-admin -n kube-system
# 用户授权
kubectl create clusterrolebinding dashboard-admin --clusterrole=cluster-admin --serviceaccount=kube-system:dashboard-admin
# 获取用户Token
kubectl describe secrets -n kube-system $(kubectl -n kube-system get secret | awk '/dashboard-admin/{print $1}')
其他常用资源接口类实例化:
core_api = client.CoreV1Api() # namespace,pod,service,pv,pvc
apps_api = client.AppsV1Api() # deployment
networking_api = client.NetworkingV1beta1Api() # ingress
storage_api = client.StorageV1Api() # storage_class
Deployment操作:
# 查询
for dp in apps_api.list_deployment_for_all_namespaces().items:
print(dp.metadata.name)
# 创建
namespace = "default"
name = "api-test"
replicas = 3
labels = {'a':'1', 'b':'2'} # 不区分数据类型,都要加引号
image = "nginx"
body = client.V1Deployment(
api_version="apps/v1",
kind="Deployment",
metadata=client.V1ObjectMeta(name=name),
spec=client.V1DeploymentSpec(
replicas=replicas,
selector={'matchLabels': labels},
template=client.V1PodTemplateSpec(
metadata=client.V1ObjectMeta(labels=labels),
spec=client.V1PodSpec(
containers=[client.V1Container(
name="web",
image=image
)]
)
),
)
)
try:
apps_api.create_namespaced_deployment(namespace=namespace, body=body)
except Exception as e:
status = getattr(e, "status")
if status == 400:
print(e)
print("格式错误")
elif status == 403:
print("没权限")
# 删除
namespace = "default"
name = "api-test"
apps_api.delete_namespaced_deployment(namespace=namespace, name=name)
Service操作:
# 查询
for svc in core_api.list_namespaced_service(namespace="default").items:
print(svc.metadata.name)
# 创建
core_api = client.CoreV1Api()
namespace = "default"
name = "api-test"
selector = {'a':'1', 'b':'2'} # 不区分数据类型,都要加引号
port = 80
target_port = 80
type = "NodePort"
body = client.V1Service(
api_version="v1",
kind="Service",
metadata=client.V1ObjectMeta(
name=name
),
spec=client.V1ServiceSpec(
selector=selector,
ports=[client.V1ServicePort(
port=port,
target_port=target_port
)],
type=type
)
)
core_api.create_namespaced_service(namespace=namespace, body=body)
# 删除
core_api.delete_namespaced_service(namespace=namespace, name=name)
直接通过HTTP客户端访问K8s HTTP API接口:
token="eyJhbGciOiJSUzI1NiIsI..."
curl --cacert /etc/kubernetes/pki/ca.crt -H "Authorization: Bearer $token" https://192.168.31.71:6443/api/v1/namespaces/default/pods
curl https://192.168.31.71:6443/api/v1/nodes \
--cacert /etc/kubernetes/pki/ca.crt \
--cert /etc/kubernetes/pki/apiserver-kubelet-client.crt \
--key /etc/kubernetes/pki/apiserver-kubelet-client.key
创建应用,用于存放首页、登录、退出、命名空间等公共资源。
python manage.py startapp dashboard
[图片上传失败...(image-e443fc-1648547221890)]
使用session就必须启用数据库,用于保存session状态:
python manage.py makemigrations
python manage.py migrate
登录认证流程:AJAX提交登录认证数据->验证提交数据格式合法性(编写一个登录认证检查函数)->确认没问题向session里写入认证信息->返回AJAX,AJAX跳转到首页。
登录认证检查:
import os
from kubernetes import client, config
import yaml
from dashboard.models import User
# 验证认证信息是否有效
def auth_check(auth_type, token):
if auth_type == "token":
configuration = client.Configuration()
configuration.host = "https://192.168.31.71:6443"
# configuration.ssl_ca_cert = os.path.join('kubeconfig', 'ca.crt')
configuration.verify_ssl = False
configuration.api_key = {"authorization": "Bearer " + token}
client.Configuration.set_default(configuration)
try:
core_api = client.CoreApi()
core_api.get_api_versions() # 随便查询个资源测试
return True
except Exception as e:
print(e)
return False
elif auth_type == "kubeconfig":
try:
user= User.objects.filter(token=token)
content = user[0].content
content = yaml.load(content, Loader=yaml.FullLoader)
config.load_kube_config_from_dict(content)
core_api = client.CoreApi()
core_api.get_api_versions()
return True
except Exception as e:
print(e)
return False
登录认证装饰器:
def self_login_required(func):
def inner(request, *args, **kwargs):
is_login = request.session.get('is_login', False)
if is_login:
return func(request, *args, **kwargs)
else:
return redirect("/login")
return inner
视图:
from django.shortcuts import render
from django.http import JsonResponse
from kubernetes import client, config
import os,hashlib,random
from devops import k8s
# Create your views here.
@k8s.self_login_required
def index(request):
return render(request, 'index.html')
def login(request):
if request.method == "GET":
return render(request, 'login.html')
elif request.method == "POST":
print(request.POST)
token = request.POST.get("token")
# 处理token登录
if token:
# 验证你的token是不是有效,对于k8s来说这个token能不能用,如果能用跳转到首页
if k8s.auth_check('token', token):
request.session['is_login'] = True
request.session['auth_type'] = "token"
request.session['token'] = token # 还需要标识用户,之前我们写的是用户名
code = 0
msg = "登录成功"
else:
code = 1
msg = "Token无效!"
else:
# 处理kubeconfig文件登录
file_obj = request.FILES.get("file")
# 生成一个随时字符串(token)保存到session中作为kubeconfig登录标识用户
token_random = hashlib.md5(str(random.random()).encode()).hexdigest()
try:
content = file_obj.read().decode() # bytes to str
User.objects.create(
auth_type="kubeconfig",
token=token_random,
content=content
)
except Exception:
code = 1
msg = "文件类型错误!"
if k8s.auth_check('kubeconfig', token_random):
request.session['is_login'] = True
request.session['auth_type'] = "kubeconfig"
request.session['token'] = token_random # 标识kubeconfig登录用户
code = 0
msg = "登录成功"
else:
code = 1
msg = "kubeconfig文件无效!"
result = {'code': code, 'msg': msg}
return JsonResponse(result)
根据布局创建对象的django 应用。
导航栏显示:
layui-nav-itemed # 展开的样式类
layui-this # 子菜单选中背景样式类
<li class="layui-nav-item {% block nav-item-1 %}{% endblock %}">
<a href="javascript:;">Kubernetes</a>
<dl class="layui-nav-child">
<dd><a href="{% url 'node' %}" class="{% block nav-this-1-1 %}{% endblock %}">Nodes</a></dd>
<dd><a href="{% url 'namespace' %}" class="{% block nav-this-1-2 %}{% endblock %}">Namespaces</a></dd>
<dd><a href="javascript:;" class="{% block nav-this-1-3 %}{% endblock %}">PersistentVolumes</a></dd>
</dl>
</li>
工作流程:登录成功跳转到首页->ajax GET请求namespace接口获取所有命名空间并追究到select列表项,再设置default为默认命名空间->再将当前命名空间存储到本地session,实现其他页面能共享获取,每次根据当前命名空间请求资源接口。
大致思路:
使用Layui从接口获取JSON数据,动态渲染表格
Django准备接口,以JSON格式返回
资源增删改查采用不同HTTP方法:
| HTTP方法 | 数据处理 | 说明 |
|---|---|---|
| POST | 新增 | 新增一个资源 |
| GET | 获取 | 取得一个资源 |
| PUT | 更新 | 更新一个资源 |
| DELETE | 删除 | 删除一个资源 |
查询:
for ns in core_api.list_namespace().items:
name = ns.metadata.name
labels = ns.metadata.labels
create_time = ns.metadata.creation_timestamp
namespace = {"name": name, "labels": labels, "create_time": create_time}
字段:名称、标签、创建时间
删除:
core_api.delete_namespace(name=name)
创建:
body = client.V1Namespace(
api_version="v1",
kind="Namespace",
metadata=client.V1ObjectMeta(
name=ns_name
)
)
core_api.create_namespace(body=body)
数据表格:
table.render({
elem: '#test'
,url:'{% url 'namespace_api' %}'
,toolbar: '#toolbarDemo' //开启头部工具栏,并为其绑定左侧模板
,defaultToolbar: ['filter', 'exports', 'print', { //自定义头部工具栏右侧图标。如无需自定义,去除该参数即可
title: '提示'
,layEvent: 'LAYTABLE_TIPS'
,icon: 'layui-icon-tips'
}]
,cols: [[
{field: 'name', title: '名称', sort: true}
,{field: 'labels', title: '标签',templet: labelsFormat}
,{field: 'create_time', title: '创建时间'}
,{fixed: 'right', title:'操作', toolbar: '#barDemo', width:150}
]]
,page: true
});
// 标签格式化,是一个对象
function labelsFormat(d){
result = "";
if (d.labels == null){
return "None"
} else {
for (let key in d.labels) {
result += '<span style="border: 1px solid #d6e5ec;border-radius: 8px">' +
key + ':' + d.labels[key] +
'</span><br>'
}
return result
}
}
查询:
for node in core_api.list_node_with_http_info()[0].items:
name = node.metadata.name
labels = node.metadata.labels
status = node.status.conditions[-1].status
scheduler = ("是" if node.spec.unschedulable is None else "否")
cpu = node.status.capacity['cpu']
memory = node.status.capacity['memory']
kebelet_version = node.status.node_info.kubelet_version
cri_version = node.status.node_info.container_runtime_version
create_time = node.metadata.creation_timestamp
node = {"name": name, "labels": labels, "status":status,
"scheduler":scheduler , "cpu":cpu, "memory":memory,
"kebelet_version":kebelet_version, "cri_version":cri_version,
"create_time": create_time}
字段:名称、标签、准备就绪、可调度、CPU、内存、kubelet版本、CRI版本、创建时间
数据表格:
table.render({
elem: '#test'
,url:'{% url 'node_api' %}'
,toolbar: '#toolbarDemo' //开启头部工具栏,并为其绑定左侧模板
,defaultToolbar: ['filter', 'exports', 'print', { //自定义头部工具栏右侧图标。如无需自定义,去除该参数即可
title: '提示'
,layEvent: 'LAYTABLE_TIPS'
,icon: 'layui-icon-tips'
}]
,cols: [[
{field: 'name', title: '名称', sort: true}
,{field: 'labels', title: '标签',templet: labelsFormat}
,{field: 'status', title: '准备就绪'}
,{field: 'scheduler', title: '可调度'}
,{field: 'cpu', title: 'CPU'}
,{field: 'memory', title: '内存'}
,{field: 'kebelet_version', title: 'kubelet版本'}
,{field: 'cri_version', title: 'CRI版本'}
,{field: 'create_time', title: '创建时间'}
,{fixed: 'right', title:'操作', toolbar: '#barDemo', width:150}
]]
,page: true
});
// 标签格式化,是一个对象
function labelsFormat(d){
result = "";
if (d.labels == null){
return "None"
} else {
for (let key in d.labels) {
result += '<span style="border: 1px solid #d6e5ec;border-radius: 8px">' +
key + ':' + d.labels[key] +
'</span><br>'
}
return result
}
}
查询:
for pv in core_api.list_persistent_volume().items:
name = pv.metadata.name
capacity = pv.spec.capacity["storage"]
access_modes = pv.spec.access_modes
reclaim_policy = pv.spec.persistent_volume_reclaim_policy
status = pv.status.phase
if pv.spec.claim_ref is not None:
pvc_ns = pv.spec.claim_ref.namespace
pvc_name = pv.spec.claim_ref.name
pvc = "%s / %s" % (pvc_ns, pvc_name)
else:
pvc = "未绑定"
storage_class = pv.spec.storage_class_name
create_time = pv.metadata.creation_timestamp
pv = {"name": name, "capacity": capacity, "access_modes":access_modes,
"reclaim_policy":reclaim_policy , "status":status, "pvc":pvc,
"storage_class":storage_class,"create_time": create_time}
字段:名称、容量、访问模式、回收策略、状态、卷申请(PVC)/命名空间、存储类、创建时间
创建:
name = request.POST.get("name", None)
capacity = request.POST.get("capacity", None)
access_mode = request.POST.get("access_mode", None)
storage_type = request.POST.get("storage_type", None)
server_ip = request.POST.get("server_ip", None)
mount_path = request.POST.get("mount_path", None)
body = client.V1PersistentVolume(
api_version="v1",
kind="PersistentVolume",
metadata=client.V1ObjectMeta(name=name),
spec=client.V1PersistentVolumeSpec(
capacity={'storage':capacity},
access_modes=[access_mode],
nfs=client.V1NFSVolumeSource(
server=server_ip,
path="/ifs/kubernetes/%s" %mount_path
)
)
)
core_api.create_persistent_volume(body=body)
删除:
core_api.delete_persistent_volume(name=name)
数据表格:
table.render({
elem: '#test'
,url:'{% url 'pv_api' %}'
,toolbar: '#toolbarDemo' //开启头部工具栏,并为其绑定左侧模板
,defaultToolbar: ['filter', 'exports', 'print', { //自定义头部工具栏右侧图标。如无需自定义,去除该参数即可
title: '提示'
,layEvent: 'LAYTABLE_TIPS'
,icon: 'layui-icon-tips'
}]
,cols: [[
{field: 'name', title: '名称', sort: true}
,{field: 'capacity', title: '容量'}
,{field: 'access_modes', title: '访问模式'}
,{field: 'reclaim_policy', title: '回收策略'}
,{field: 'status', title: '状态'}
,{field: 'pvc', title: 'PVC(命名空间/名称)'}
,{field: 'storage_class', title: '存储类'}
,{field: 'create_time', title: '创建时间'}
,{fixed: 'right', title:'操作', toolbar: '#barDemo', width:150}
]]
,page: true
,id: 'pvtb'
});
查询:
for dp in apps_api.list_namespaced_deployment(namespace).items:
name = dp.metadata.name
namespace = dp.metadata.namespace
replicas = dp.spec.replicas
available_replicas = ( 0 if dp.status.available_replicas is None else dp.status.available_replicas)
labels = dp.metadata.labels
selector = dp.spec.selector.match_labels
containers = {}
for c in dp.spec.template.spec.containers:
containers[c.name] = c.image
create_time = dp.metadata.creation_timestamp
dp = {"name": name, "namespace": namespace, "replicas":replicas,
"available_replicas":available_replicas , "labels":labels, "selector":selector,
"containers":containers, "create_time": create_time}
字段:名称、命名空间、预期副本数、可用副本数、Pod标签选择器、镜像/状态、创建时间
创建:
name = request.POST.get("name",None)
namespace = request.POST.get("namespace",None)
image = request.POST.get("image",None)
replicas = int(request.POST.get("replicas",None))
# 处理标签
labels = {}
try:
for l in request.POST.get("labels",None).split(","):
k = l.split("=")[0]
v = l.split("=")[1]
labels[k] = v
except Exception as e:
res = {"code": 1, "msg": "标签格式错误!"}
return JsonResponse(res)
resources = request.POST.get("resources",None)
health_liveness = request.POST.get("health[liveness]",None) # {'health[liveness]': ['on'], 'health[readiness]': ['on']}
health_readiness = request.POST.get("health[readiness]",None)
if resources == "1c2g":
resources = client.V1ResourceRequirements(limits={"cpu":"1","memory":"1Gi"},
requests={"cpu":"0.9","memory":"0.9Gi"})
elif resources == "2c4g":
resources = client.V1ResourceRequirements(limits={"cpu": "2", "memory": "4Gi"},
requests={"cpu": "1.9", "memory": "3.9Gi"})
elif resources == "4c8g":
resources = client.V1ResourceRequirements(limits={"cpu": "4", "memory": "8Gi"},
requests={"cpu": "3.9", "memory": "7.9Gi"})
else:
resources = client.V1ResourceRequirements(limits={"cpu":"500m","memory":"1Gi"},
requests={"cpu":"450m","memory":"900Mi"})
liveness_probe = ""
if health_liveness == "on":
liveness_probe = client.V1Probe(http_get="/",timeout_seconds=30,initial_delay_seconds=30)
readiness_probe = ""
if health_readiness == "on":
readiness_probe = client.V1Probe(http_get="/",timeout_seconds=30,initial_delay_seconds=30)
for dp in apps_api.list_namespaced_deployment(namespace=namespace).items:
if name == dp.metadata.name:
res = {"code": 1, "msg": "Deployment已经存在!"}
return JsonResponse(res)
body = client.V1Deployment(
api_version="apps/v1",
kind="Deployment",
metadata=client.V1ObjectMeta(name=name),
spec=client.V1DeploymentSpec(
replicas=replicas,
selector={'matchLabels': labels},
template=client.V1PodTemplateSpec(
metadata=client.V1ObjectMeta(labels=labels),
spec=client.V1PodSpec(
containers=[client.V1Container( # https://github.com/kubernetes-client/python/blob/master/kubernetes/docs/V1Container.md
name="web",
image=image,
env=[{"name": "TEST", "value": "123"}, {"name": "DEV", "value": "456"}],
ports=[client.V1ContainerPort(container_port=80)],
# liveness_probe=liveness_probe,
# readiness_probe=readiness_probe,
resources=resources,
)]
)
),
)
)
apps_api.create_namespaced_deployment(namespace=namespace, body=body)
删除:
apps_api.delete_namespaced_deployment(namespace=namespace, name=name)
数据表格:
table.render({
elem: '#test'
,url:'{% url 'deployment_api' %}?namespace=' + namespace
,toolbar: '#toolbarDemo' //开启头部工具栏,并为其绑定左侧模板
,defaultToolbar: ['filter', 'exports', 'print', { //自定义头部工具栏右侧图标。如无需自定义,去除该参数即可
title: '提示'
,layEvent: 'LAYTABLE_TIPS'
,icon: 'layui-icon-tips'
}]
,cols: [[
{field: 'name', title: '名称', sort: true}
,{field: 'namespace', title: '命名空间'}
,{field: 'replicas', title: '预期副本数'}
,{field: 'available_replicas', title: '可用副本数'}
,{field: 'labels', title: '标签',templet: labelsFormat}
,{field: 'selector', title: 'Pod标签选择器',templet: selecotrFormat}
,{field: 'containers', title: '容器', templet: containersFormat}
,{field: 'create_time', title: '创建时间'}
,{fixed: 'right', title:'操作', toolbar: '#barDemo', width:150}
]]
,page: true
,id: 'dptb'
});
// 标签格式化,是一个对象
function labelsFormat(d){
result = "";
if(d.labels == null){
return "None"
} else {
for (let key in d.labels) {
result += '<span style="border: 1px solid #d6e5ec;border-radius: 8px">' +
key + ':' + d.labels[key] +
'</span><br>'
}
return result
}
}
function selecotrFormat(d){
result = "";
for(let key in d.selector) {
result += '<span style="border: 1px solid #d6e5ec;border-radius: 8px">' +
key + ':' + d.selector[key] +
'</span><br>'
}
return result
}
function containersFormat(d) {
result = "";
for(let key in d.containers) {
result += key + '=' + d.containers[key] + '<br>'
}
return result
}
查询:
for ds in apps_api.list_namespaced_daemon_set(namespace).items:
name = ds.metadata.name
namespace = ds.metadata.namespace
desired_number = ds.status.desired_number_scheduled
available_number = ds.status.number_available
labels = ds.metadata.labels
selector = ds.spec.selector.match_labels
containers = {}
for c in ds.spec.template.spec.containers:
containers[c.name] = c.image
create_time = ds.metadata.creation_timestamp
ds = {"name": name, "namespace": namespace, "labels": labels, "desired_number": desired_number,
"available_number": available_number,
"selector": selector, "containers": containers, "create_time": create_time}
字段:名称、命名空间、预期节点数、可用节点数、Pod标签选择器、镜像、创建时间
创建:
name = request.POST.get("name",None)
namespace = request.POST.get("namespace",None)
image = request.POST.get("image",None)
# 处理标签
labels = {}
try:
for l in request.POST.get("labels",None).split(","):
k = l.split("=")[0]
v = l.split("=")[1]
labels[k] = v
except Exception as e:
res = {"code": 1, "msg": "标签格式错误!"}
return JsonResponse(res)
resources = request.POST.get("resources",None)
health_liveness = request.POST.get("health[liveness]",None) # {'health[liveness]': ['on'], 'health[readiness]': ['on']}
health_readiness = request.POST.get("health[readiness]",None)
if resources == "1c2g":
resources = client.V1ResourceRequirements(limits={"cpu":"1","memory":"1Gi"},
requests={"cpu":"0.9","memory":"0.9Gi"})
elif resources == "2c4g":
resources = client.V1ResourceRequirements(limits={"cpu": "2", "memory": "4Gi"},
requests={"cpu": "1.9", "memory": "3.9Gi"})
elif resources == "4c8g":
resources = client.V1ResourceRequirements(limits={"cpu": "4", "memory": "8Gi"},
requests={"cpu": "3.9", "memory": "7.9Gi"})
else:
resources = client.V1ResourceRequirements(limits={"cpu":"500m","memory":"1Gi"},
requests={"cpu":"450m","memory":"900Mi"})
liveness_probe = ""
if health_liveness == "on":
liveness_probe = client.V1Probe(http_get="/",timeout_seconds=30,initial_delay_seconds=30)
readiness_probe = ""
if health_readiness == "on":
readiness_probe = client.V1Probe(http_get="/",timeout_seconds=30,initial_delay_seconds=30)
for dp in apps_api.list_namespaced_daemon_set(namespace=namespace).items:
if name == dp.metadata.name:
res = {"code": 1, "msg": "DaemonSet已经存在!"}
return JsonResponse(res)
body = client.V1DaemonSet(
api_version="apps/v1",
kind="DaemonSet",
metadata=client.V1ObjectMeta(name=name),
spec=client.V1DeploymentSpec(
selector={'matchLabels': labels},
template=client.V1PodTemplateSpec(
metadata=client.V1ObjectMeta(labels=labels),
spec=client.V1PodSpec(
containers=[client.V1Container( # https://github.com/kubernetes-client/python/blob/master/kubernetes/docs/V1Container.md
name="web",
image=image,
env=[{"name": "TEST", "value": "123"}, {"name": "DEV", "value": "456"}],
ports=[client.V1ContainerPort(container_port=80)],
# liveness_probe=liveness_probe,
# readiness_probe=readiness_probe,
resources=resources,
)]
)
),
)
)
删除:
apps_api.delete_namespaced_daemon_set(namespace=namespace, name=name)
数据表格:
table.render({
elem: '#test'
,url:'{% url 'daemonset_api' %}?namespace=' + namespace
,toolbar: '#toolbarDemo' //开启头部工具栏,并为其绑定左侧模板
,defaultToolbar: ['filter', 'exports', 'print', { //自定义头部工具栏右侧图标。如无需自定义,去除该参数即可
title: '提示'
,layEvent: 'LAYTABLE_TIPS'
,icon: 'layui-icon-tips'
}]
,cols: [[
{field: 'name', title: '名称', sort: true}
,{field: 'namespace', title: '命名空间',sort: true}
,{field: 'desired_number', title: '预期节点数',width: 100}
,{field: 'available_number', title: '可用节点数',width: 100}
,{field: 'labels', title: '标签',templet: labelsFormat}
,{field: 'selector', title: 'Pod 标签选择器',templet: selecotrFormat}
,{field: 'containers', title: '容器', templet: containersFormat}
,{field: 'create_time', title: '创建时间',width: 200}
,{fixed: 'right', title:'操作', toolbar: '#barDemo',width: 150}
]]
,page: true
,id: 'dstb'
});
// 标签格式化,是一个对象
function labelsFormat(d){
result = "";
if(d.labels == null){
return "None"
} else {
for (let key in d.labels) {
result += '<span style="border: 1px solid #d6e5ec;border-radius: 8px">' +
key + ':' + d.labels[key] +
'</span><br>'
}
return result
}
}
function selecotrFormat(d){
result = "";
for(let key in d.selector) {
result += '<span style="border: 1px solid #d6e5ec;border-radius: 8px">' +
key + ':' + d.selector[key] +
'</span><br>'
}
return result
}
function containersFormat(d) {
result = "";
for(let key in d.containers) {
result += key + '=' + d.containers[key] + '<br>'
}
return result
}
查询:
for sts in apps_api.list_namespaced_stateful_set(namespace).items:
name = sts.metadata.name
namespace = sts.metadata.namespace
labels = sts.metadata.labels
selector = sts.spec.selector.match_labels
replicas = sts.spec.replicas
ready_replicas = ("0" if sts.status.ready_replicas is None else sts.status.ready_replicas)
#current_replicas = sts.status.current_replicas
service_name = sts.spec.service_name
containers = {}
for c in sts.spec.template.spec.containers:
containers[c.name] = c.image
create_time = k8s.timestamp_format(sts.metadata.creation_timestamp)
ds = {"name": name, "namespace": namespace, "labels": labels, "replicas": replicas,
"ready_replicas": ready_replicas, "service_name": service_name,
"selector": selector, "containers": containers, "create_time": create_time}
字段:名称、命名空间、Service名称、预期副本数、可用副本数、Pod标签选择器、镜像、创建时间
删除:
apps_api.delete_namespaced_stateful_set(namespace=namespace, name=name)
数据表格:
table.render({
elem: '#test'
,url:'{% url 'statefulset_api' %}?namespace=' + namespace
,toolbar: '#toolbarDemo' //开启头部工具栏,并为其绑定左侧模板
,defaultToolbar: ['filter', 'exports', 'print', { //自定义头部工具栏右侧图标。如无需自定义,去除该参数即可
title: '提示'
,layEvent: 'LAYTABLE_TIPS'
,icon: 'layui-icon-tips'
}]
,cols: [[
{field: 'name', title: '名称', sort: true}
,{field: 'namespace', title: '命名空间',sort: true}
,{field: 'service_name', title: 'Service名称'}
,{field: 'replicas', title: '预期副本数',width: 100}
,{field: 'ready_replicas', title: '可用副本数',width: 100}
,{field: 'labels', title: '标签',templet: labelsFormat}
,{field: 'selector', title: 'Pod 标签选择器',templet: selecotrFormat}
,{field: 'containers', title: '容器', templet: containersFormat}
,{field: 'create_time', title: '创建时间',width: 200}
,{fixed: 'right', title:'操作', toolbar: '#barDemo',width: 150}
]]
,page: true
,id: 'ststb'
});
// 标签格式化,是一个对象
function labelsFormat(d){
result = "";
if(d.labels == null){
return "None"
} else {
for (let key in d.labels) {
result += '<span style="border: 1px solid #d6e5ec;border-radius: 8px">' +
key + ':' + d.labels[key] +
'</span><br>'
}
return result
}
}
function selecotrFormat(d){
result = "";
for(let key in d.selector) {
result += '<span style="border: 1px solid #d6e5ec;border-radius: 8px">' +
key + ':' + d.selector[key] +
'</span><br>'
}
return result
}
function containersFormat(d) {
result = "";
for(let key in d.containers) {
result += key + '=' + d.containers[key] + '<br>'
}
return result
}
查询:
for po in core_api.list_namespaced_pod(namespace).items:
name = po.metadata.name
namespace = po.metadata.namespace
labels = po.metadata.labels
pod_ip = po.status.pod_ip
containers = [] # [{},{},{}]
status = "None"
# 只为None说明Pod没有创建(不能调度或者正在下载镜像)
if po.status.container_statuses is None:
status = po.status.conditions[-1].reason
else:
for c in po.status.container_statuses:
c_name = c.name
c_image = c.image
# 获取重启次数
restart_count = c.restart_count
# 获取容器状态
c_status = "None"
if c.ready is True:
c_status = "Running"
elif c.ready is False:
if c.state.waiting is not None:
c_status = c.state.waiting.reason
elif c.state.terminated is not None:
c_status = c.state.terminated.reason
elif c.state.last_state.terminated is not None:
c_status = c.last_state.terminated.reason
c = {'c_name': c_name,'c_image':c_image ,'restart_count': restart_count, 'c_status': c_status}
containers.append(c)
create_time = k8s.timestamp_format(po.metadata.creation_timestamp)
po = {"name": name, "namespace": namespace, "pod_ip": pod_ip,
"labels": labels, "containers": containers, "status": status,
"create_time": create_time}
字段:名称、命名空间、IP地址、标签、容器组、状态、创建时间
删除:
core_api.delete_namespaced_pod(namespace=namespace, name=name)
数据表格:
table.render({
elem: '#test'
,url:'{% url 'pod_api' %}?namespace=' + namespace
,toolbar: '#toolbarDemo' //开启头部工具栏,并为其绑定左侧模板
,defaultToolbar: ['filter', 'exports', 'print', { //自定义头部工具栏右侧图标。如无需自定义,去除该参数即可
title: '提示'
,layEvent: 'LAYTABLE_TIPS'
,icon: 'layui-icon-tips'
}]
,cols: [[
{field: 'name', title: '名称', sort: true}
,{field: 'namespace', title: '命名空间',sort: true}
,{field: 'pod_ip', title: 'IP地址'}
,{field: 'labels', title: '标签', templet: labelsFormat}
,{field: 'containers', title: '容器组', templet: containersFormat}
,{field: 'status', title: '状态',sort: true, templet: statusFormat}
,{field: 'create_time', title: '创建时间'}
,{fixed: 'right', title:'操作', toolbar: '#barDemo',width: 250}
]]
,page: true
,id: 'potb'
});
// 标签格式化,是一个对象
function labelsFormat(d){
result = "";
if(d.labels == null){
return "None"
} else {
for (let key in d.labels) {
result += '<span style="border: 1px solid #d6e5ec;border-radius: 8px">' +
key + ':' + d.labels[key] +
'</span><br>'
}
return result
}
}
function containersFormat(d) {
if (d.containers) {
for(let key in d.containers) {
data = d.containers[key];
result += key + ':' + data.c_name + '=' + data.c_image + '<br>' +
'重启次数:' + data.restart_count + '<br>' +
'状态:' + data.c_status + '<br>'
}
return result
} else {
return "None"
}
}
// 如果status为None,使用容器状态显示
function statusFormat(d){
if(d.status == "None"){
for(let key in d.containers) {
result += d.containers[key].c_status + '<br>'
}
return result
} else {
return d.status
}
}
查询:
for svc in core_api.list_namespaced_service(namespace=namespace).items:
name = svc.metadata.name
namespace = svc.metadata.namespace
labels = svc.metadata.labels
type = svc.spec.type
cluster_ip = svc.spec.cluster_ip
ports = []
for p in svc.spec.ports: # 不是序列,不能直接返回
port_name = p.name
port = p.port
target_port = p.target_port
protocol = p.protocol
node_port = ""
if type == "NodePort":
node_port = " <br> NodePort: %s" % p.node_port
port = {'port_name': port_name, 'port': port, 'protocol': protocol, 'target_port':target_port, 'node_port': node_port}
ports.append(port)
selector = svc.spec.selector
create_time = svc.metadata.creation_timestamp
# 确认是否关联Pod
endpoint = ""
for ep in core_api.list_namespaced_endpoints(namespace=namespace).items:
if ep.metadata.name == name and ep.subsets is None:
endpoint = "未关联"
else:
endpoint = "已关联"
svc = {"name": name, "namespace": namespace, "type": type,
"cluster_ip": cluster_ip, "ports": ports, "labels": labels,
"selector": selector, "endpoint": endpoint, "create_time": create_time}
字段:名称、命名空间、类型、集群IP、端口信息、Pod标签选择器、后端Pod、创建时间
创建:
name = request.POST.get("name",None)
namespace = request.POST.get("namespace",None)
port = int(request.POST.get("port",None))
target_port = int(request.POST.get("target-port",None))
labels = {}
try:
for l in request.POST.get("labels",None).split(","):
k = l.split("=")[0]
v = l.split("=")[1]
labels[k] = v
except Exception as e:
res = {"code": 1, "msg": "标签格式错误!"}
return JsonResponse(res)
type = request.POST.get("type","")
body = client.V1Service(
api_version="v1",
kind="Service",
metadata=client.V1ObjectMeta(
name=name
),
spec=client.V1ServiceSpec(
selector=labels,
ports=[client.V1ServicePort(
port=port,
target_port=target_port,
)],
type=type
)
)
core_api.create_namespaced_service(namespace=namespace, body=body)
删除:
core_api.delete_namespaced_service(namespace=namespace, name=name)
数据表格:
table.render({
elem: '#test'
,url:'{% url 'service_api' %}?namespace=' + namespace
,toolbar: '#toolbarDemo' //开启头部工具栏,并为其绑定左侧模板
,defaultToolbar: ['filter', 'exports', 'print', { //自定义头部工具栏右侧图标。如无需自定义,去除该参数即可
title: '提示'
,layEvent: 'LAYTABLE_TIPS'
,icon: 'layui-icon-tips'
}]
,cols: [[
{field: 'name', title: '名称', sort: true, width: 300}
,{field: 'namespace', title: '命名空间',width: 200, sort: true}
,{field: 'type', title: '类型',width: 120, sort: true}
,{field: 'cluster_ip', title: '集群IP',width: 150}
,{field: 'ports', title: '端口信息',templet: portsFormat}
,{field: 'labels', title: '标签', templet: labelsFormat}
,{field: 'selector', title: 'Pod 标签选择器', templet: selecotrFormat}
,{field: 'endpoint', title: '后端 Pod'}
,{field: 'create_time', title: '创建时间',width: 200}
,{fixed: 'right', title:'操作', toolbar: '#barDemo',width: 150}
]]
,page: true
,id: 'svctb'
});
// 标签格式化,是一个对象
function labelsFormat(d){
result = "";
if(d.labels == null){
return "None"
} else {
for (let key in d.labels) {
result += '<span style="border: 1px solid #d6e5ec;border-radius: 8px">' +
key + ':' + d.labels[key] +
'</span><br>'
}
return result
}
}
function selecotrFormat(d){
result = "";
for(let key in d.selector) {
result += '<span style="border: 1px solid #d6e5ec;border-radius: 8px">' +
key + ':' + d.selector[key] +
'</span><br>'
}
return result
}
function portsFormat(d) {
result = "";
for(let key in d.ports) {
data = d.ports[key];
result += '名称: ' + data.port_name + '<br>' +
'端口: ' + data.port + '<br>' +
'协议: ' + data.protocol + '<br>' +
'容器端口: ' + data.target_port + '<br>'
}
return result
}
查询:
for ing in networking_api.list_namespaced_ingress(namespace=namespace).items:
name = ing.metadata.name
namespace = ing.metadata.namespace
labels = ing.metadata.labels
service = "None"
http_hosts = "None"
for h in ing.spec.rules:
host = h.host
path = ("/" if h.http.paths[0].path is None else h.http.paths[0].path)
service_name = h.http.paths[0].backend.service_name
service_port = h.http.paths[0].backend.service_port
http_hosts = {'host': host, 'path': path, 'service_name': service_name, 'service_port': service_port}
https_hosts = "None"
if ing.spec.tls is None:
https_hosts = ing.spec.tls
else:
for tls in ing.spec.tls:
host = tls.hosts[0]
secret_name = tls.secret_name
https_hosts = {'host': host, 'secret_name': secret_name}
create_time = k8s.timestamp_format(ing.metadata.creation_timestamp)
ing = {"name": name, "namespace": namespace,"labels": labels ,"http_hosts": http_hosts,
"https_hosts": https_hosts, "service": service, "create_time": create_time
字段:名称、命名空间、HTTP、HTTPS、关联Service、创建时间
创建:
name = request.POST.get("name",None)
namespace = request.POST.get("namespace",None)
host = request.POST.get("host",None)
path = request.POST.get("path","/")
svc_name = request.POST.get("svc_name",None)
svc_port = int(request.POST.get("svc_port",None))
body = client.NetworkingV1beta1Ingress(
api_version="networking.k8s.io/v1beta1",
kind="Ingress",
metadata=client.V1ObjectMeta(name=name, annotations={
"nginx.ingress.kubernetes.io/rewrite-target": "/"
}),
spec=client.NetworkingV1beta1IngressSpec(
rules=[client.NetworkingV1beta1IngressRule(
host=host,
http=client.NetworkingV1beta1HTTPIngressRuleValue(
paths=[client.NetworkingV1beta1HTTPIngressPath(
path=path,
backend=client.NetworkingV1beta1IngressBackend(
service_port=svc_port,
service_name=svc_name)
)]
)
)
]
)
)
networking_api.create_namespaced_ingress(namespace=namespace, body=body)
删除:
networking_api.delete_namespaced_ingress(namespace=namespace, name=name)
数据表格:
table.render({
elem: '#test'
,url:'{% url 'ingress_api' %}?namespace=' + namespace
,toolbar: '#toolbarDemo' //开启头部工具栏,并为其绑定左侧模板
,defaultToolbar: ['filter', 'exports', 'print', { //自定义头部工具栏右侧图标。如无需自定义,去除该参数即可
title: '提示'
,layEvent: 'LAYTABLE_TIPS'
,icon: 'layui-icon-tips'
}]
,cols: [[
{field: 'name', title: '名称', sort: true, width: 300}
,{field: 'namespace', title: '命名空间',width: 200, sort: true}
,{field: 'http_hosts', title: 'HTTP',templet: httpFormat}
,{field: 'https_hosts', title: 'HTTPS',templet: httpsFormat}
,{field: 'service', title: '关联 Service', templet: serviceFormat}
,{field: 'create_time', title: '创建时间',width: 200}
,{fixed: 'right', title:'操作', toolbar: '#barDemo',width: 150}
]]
,page: true
,id: 'ingtb'
});
// 标签格式化,是一个对象
function httpFormat(d){
return "域名: " + d.http_hosts.host + '<br>' + "路径: " + d.http_hosts.path + '<br>'
}
function httpsFormat(d){
if(d.https_hosts != null){
return "域名: " + d.https_hosts.host + '<br>' + "证书Secret名称: " + d.https_hosts.secret_name + '<br>';
} else {
return "None"
}
}
function serviceFormat(d) {
return "名称: " + d.http_hosts.service_name + '<br>' + "端口: " + d.http_hosts.service_port + '<br>';
}
查询:
for pvc in core_api.list_namespaced_persistent_volume_claim(namespace=namespace).items:
name = pvc.metadata.name
namespace = pvc.metadata.namespace
labels = pvc.metadata.labels
storage_class_name = pvc.spec.storage_class_name
access_modes = pvc.spec.access_modes
capacity = (pvc.status.capacity if pvc.status.capacity is None else pvc.status.capacity["storage"])
volume_name = pvc.spec.volume_name
status = pvc.status.phase
create_time = pvc.metadata.creation_timestamp
pvc = {"name": name, "namespace": namespace, "lables": labels,
"storage_class_name": storage_class_name, "access_modes": access_modes, "capacity": capacity,
"volume_name": volume_name, "status": status, "create_time": create_time}
字段:名称、命名空间、状态、卷名称、容量、访问模式、存储类、创建时间
创建:
name = request.POST.get("name", None)
namespace = request.POST.get("namespace", None)
storage_class = request.POST.get("storage_class", None)
access_mode = request.POST.get("access_mode", None)
capacity = request.POST.get("capacity", None)
body = client.V1PersistentVolumeClaim(
api_version="v1",
kind="PersistentVolumeClaim",
metadata=client.V1ObjectMeta(name=name,namespace=namespace),
spec=client.V1PersistentVolumeClaimSpec(
storage_class_name=storage_class, # 使用存储类创建PV,如果不用可去掉
access_modes=[access_mode],
resources=client.V1ResourceRequirements(
requests={"storage" : capacity}
)
)
)
core_api.create_namespaced_persistent_volume_claim(namespace=namespace, body=body)
删除:
core_api.delete_namespaced_persistent_volume_claim(namespace=namespace, name=name)
数据表格:
layui.use('table', function(){
var table = layui.table;
var $ = layui.jquery;
table.render({
elem: '#test'
,url:'{% url 'pvc_api' %}?namespace=' + namespace
,toolbar: '#toolbarDemo' //开启头部工具栏,并为其绑定左侧模板
,defaultToolbar: ['filter', 'exports', 'print', { //自定义头部工具栏右侧图标。如无需自定义,去除该参数即可
title: '提示'
,layEvent: 'LAYTABLE_TIPS'
,icon: 'layui-icon-tips'
}]
,cols: [[
{field: 'name', title: '名称', sort: true}
,{field: 'namespace', title: '命名空间',sort: true}
,{field: 'labels', title: '标签',templet: labelsFormat}
,{field: 'status', title: '状态',width: 130}
,{field: 'volume_name', title: '卷名称'}
,{field: 'capacity', title: '容量',width: 130}
,{field: 'access_modes', title: '访问模式'}
,{field: 'storage_class_name', title: '存储类'}
,{field: 'create_time', title: '创建时间',width: 200}
,{fixed: 'right', title:'操作', toolbar: '#barDemo',width: 150}
]]
,page: true
,id: 'pvctb'
});
// 标签格式化,是一个对象
function labelsFormat(d){
result = "";
if(d.labels == null){
return "None"
} else {
for (let key in d.labels) {
result += '<span style="border: 1px solid #d6e5ec;border-radius: 8px">' +
key + ':' + d.labels[key] +
'</span><br>'
}
return result
}
}
查询:
for cm in core_api.list_namespaced_config_map(namespace=namespace).items:
name = cm.metadata.name
namespace = cm.metadata.namespace
data_length = ("0" if cm.data is None else len(cm.data))
create_time = cm.metadata.creation_timestamp
cm = {"name": name, "namespace": namespace, "data_length": data_length, "create_time": create_time}
字段:名称、命名空间、数据数量、创建时间
删除:
core_api.delete_namespaced_config_map(name=name,namespace=namespace)
数据表格:
table.render({
elem: '#test'
,url:'{% url 'configmap_api' %}?namespace=' + namespace
,toolbar: '#toolbarDemo' //开启头部工具栏,并为其绑定左侧模板
,defaultToolbar: ['filter', 'exports', 'print', { //自定义头部工具栏右侧图标。如无需自定义,去除该参数即可
title: '提示'
,layEvent: 'LAYTABLE_TIPS'
,icon: 'layui-icon-tips'
}]
,cols: [[
{field: 'name', title: '名称', sort: true}
,{field: 'namespace', title: '命名空间',sort: true}
,{field: 'data_length', title: '数据数量'}
,{field: 'create_time', title: '创建时间'}
,{fixed: 'right', title:'操作', toolbar: '#barDemo',width: 150}
]]
,page: true
,id: 'cmtb'
});
// 标签格式化,是一个对象
function labelsFormat(d){
result = "";
if(d.labels == null){
return "None"
} else {
for (let key in d.labels) {
result += '<span style="border: 1px solid #d6e5ec;border-radius: 8px">' +
key + ':' + d.labels[key] +
'</span><br>'
}
return result
}
}
查询:
for secret in core_api.list_namespaced_secret(namespace=namespace).items:
name = secret.metadata.name
namespace = secret.metadata.namespace
data_length = ("空" if secret.data is None else len(secret.data))
create_time = secret.metadata.creation_timestamp
se = {"name": name, "namespace": namespace, "data_length": data_length, "create_time": create_time}
字段:名称、命名空间、数据数量、创建时间
删除:
core_api.delete_namespaced_secret(namespace=namespace, name=name)
数据表格:
table.render({
elem: '#test'
,url:'{% url 'secret_api' %}?namespace=' + namespace
,toolbar: '#toolbarDemo' //开启头部工具栏,并为其绑定左侧模板
,defaultToolbar: ['filter', 'exports', 'print', { //自定义头部工具栏右侧图标。如无需自定义,去除该参数即可
title: '提示'
,layEvent: 'LAYTABLE_TIPS'
,icon: 'layui-icon-tips'
}]
,cols: [[
{field: 'name', title: '名称', sort: true}
,{field: 'namespace', title: '命名空间'}
,{field: 'data_length', title: '数据数量'}
,{field: 'create_time', title: '创建时间'}
,{fixed: 'right', title:'操作', toolbar: '#barDemo',width: 150}
]]
,page: true
,id: 'secrettb'
});
// 标签格式化,是一个对象
function labelsFormat(d){
result = "";
if(d.labels == null){
return "None"
} else {
for (let key in d.labels) {
result += '<span style="border: 1px solid #d6e5ec;border-radius: 8px">' +
key + ':' + d.labels[key] +
'</span><br>'
}
return result
}
}
1、减少视图函数认证代码
2、显示每个页面导航(面包屑)
3、每个k8s资源都有一个时间,默认是UTC,进行格式化中国时区
ACE 是一个开源的、独立的、基于浏览器的代码编辑器,可以嵌入到任何web页面或JavaScript应用程序中。
https://github.com/ajaxorg/ace
yum install epel-release -y
yum install npm git -y
git clone https://github.com/ajaxorg/ace.git
cd ace
npm install
node ./Makefile.dryice.js
使用步骤:
1、将ace包放到static目录下
2、准备导出yaml接口
3、准备ace editor编辑器页面
4、ajax调用导出的yaml接口并将yaml内容填充到档期打开的编辑器
关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。
我正在编写一个包含C扩展的gem。通常当我写一个gem时,我会遵循TDD的过程,我会写一个失败的规范,然后处理代码直到它通过,等等......在“ext/mygem/mygem.c”中我的C扩展和在gemspec的“扩展”中配置的有效extconf.rb,如何运行我的规范并仍然加载我的C扩展?当我更改C代码时,我需要采取哪些步骤来重新编译代码?这可能是个愚蠢的问题,但是从我的gem的开发源代码树中输入“bundleinstall”不会构建任何native扩展。当我手动运行rubyext/mygem/extconf.rb时,我确实得到了一个Makefile(在整个项目的根目录中),然后当
我已经在Sinatra上创建了应用程序,它代表了一个简单的API。我想在生产和开发上进行部署。我想在部署时选择,是开发还是生产,一些方法的逻辑应该改变,这取决于部署类型。是否有任何想法,如何完成以及解决此问题的一些示例。例子:我有代码get'/api/test'doreturn"Itisdev"end但是在部署到生产环境之后我想在运行/api/test之后看到ItisPROD如何实现? 最佳答案 根据SinatraDocumentation:EnvironmentscanbesetthroughtheRACK_ENVenvironm
我们的git存储库中目前有一个Gemfile。但是,有一个gem我只在我的环境中本地使用(我的团队不使用它)。为了使用它,我必须将它添加到我们的Gemfile中,但每次我checkout到我们的master/dev主分支时,由于与跟踪的gemfile冲突,我必须删除它。我想要的是类似Gemfile.local的东西,它将继承从Gemfile导入的gems,但也允许在那里导入新的gems以供使用只有我的机器。此文件将在.gitignore中被忽略。这可能吗? 最佳答案 设置BUNDLE_GEMFILE环境变量:BUNDLE_GEMFI
这似乎非常适得其反,因为太多的gem会在window上破裂。我一直在处理很多mysql和ruby-mysqlgem问题(gem本身发生段错误,一个名为UnixSocket的类显然在Windows机器上不能正常工作,等等)。我只是在浪费时间吗?我应该转向不同的脚本语言吗? 最佳答案 我在Windows上使用Ruby的经验很少,但是当我开始使用Ruby时,我是在Windows上,我的总体印象是它不是Windows原生系统。因此,在主要使用Windows多年之后,开始使用Ruby促使我切换回原来的系统Unix,这次是Linux。Rub
我正在玩HTML5视频并且在ERB中有以下片段:mp4视频从在我的开发环境中运行的服务器很好地流式传输到chrome。然而firefox显示带有海报图像的视频播放器,但带有一个大X。问题似乎是mongrel不确定ogv扩展的mime类型,并且只返回text/plain,如curl所示:$curl-Ihttp://0.0.0.0:3000/pr6.ogvHTTP/1.1200OKConnection:closeDate:Mon,19Apr201012:33:50GMTLast-Modified:Sun,18Apr201012:46:07GMTContent-Type:text/plain
这个问题在这里已经有了答案:关闭10年前。PossibleDuplicate:Pythonconditionalassignmentoperator对于这样一个简单的问题表示歉意,但是谷歌搜索||=并不是很有帮助;)Python中是否有与Ruby和Perl中的||=语句等效的语句?例如:foo="hey"foo||="what"#assignfooifit'sundefined#fooisstill"hey"bar||="yeah"#baris"yeah"另外,类似这样的东西的通用术语是什么?条件分配是我的第一个猜测,但Wikipediapage跟我想的不太一样。
什么是ruby的rack或python的Java的wsgi?还有一个路由库。 最佳答案 来自Python标准PEP333:Bycontrast,althoughJavahasjustasmanywebapplicationframeworksavailable,Java's"servlet"APImakesitpossibleforapplicationswrittenwithanyJavawebapplicationframeworktoruninanywebserverthatsupportstheservletAPI.ht
无论您是想搭建桌面端、WEB端或者移动端APP应用,HOOPSPlatform组件都可以为您提供弹性的3D集成架构,同时,由工业领域3D技术专家组成的HOOPS技术团队也能为您提供技术支持服务。如果您的客户期望有一种在多个平台(桌面/WEB/APP,而且某些客户端是“瘦”客户端)快速、方便地将数据接入到3D应用系统的解决方案,并且当访问数据时,在各个平台上的性能和用户体验保持一致,HOOPSPlatform将帮助您完成。利用HOOPSPlatform,您可以开发在任何环境下的3D基础应用架构。HOOPSPlatform可以帮您打造3D创新型产品,HOOPSSDK包含的技术有:快速且准确的CAD
华为OD机试题本篇题目:明明的随机数题目输入描述输出描述:示例1输入输出说明代码编写思路最近更新的博客华为od2023|什么是华为od,od薪资待遇,od机试题清单华为OD机试真题大全,用Python解华为机试题|机试宝典【华为OD机试】全流程解析+经验分享,题型分享,防作弊指南华为o