package main





import (

"html/template"

"io"

"io/ioutil"

"log"

"net/http"

"os"

"path"

"runtime/debug"

)





const (

ListDir      = 0x0001

UPLOAD_DIR   = "./uploads"

TEMPLATE_DIR = "./views"

)





//cache storage all template

var templates = make(map[string]*template.Template)





func init() {

fileInfoArr, err := ioutil.ReadDir(TEMPLATE_DIR)

check(err)

var templateName, templatePath string

for _, fileInfo := range fileInfoArr {

templateName = fileInfo.Name()

if ext := path.Ext(templateName); ext != ".html" {

continue

}

templatePath = TEMPLATE_DIR + "/" + templateName

//log.Println(templateName)

log.Println("Loading template:", templatePath)

t := template.Must(template.ParseFiles(templatePath))

templates[templateName] = t

}





//if const template

/*for _, tmpl := range []string{"upload", "list"} {

//Must ensure in case can't analytic will do error operate,If the template loading is not successful, the program will exit

t := template.Must(template.ParseFiles(tmpl + ".html"))

templates[tmpl] = t

}*/

}

func check(err error) {

if err != nil {

panic(err)

}

}

func renderHtml(w http.ResponseWriter, tmpl string, locals map[string]interface{}) {

/*tt := templates[tmpl]

log.Println("xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx", templates, tmpl, tt)*/

log.Println(locals)

err := templates[tmpl].Execute(w, locals)





check(err)

}

func isExists(path string) bool {

_, err := os.Stat(path)

//log.Println(err)

if err == nil {

return true

}

return os.IsExist(err)

}





//Callback method,

func uploadHandler(w http.ResponseWriter, r *http.Request) {

//log.Println(r.Method)

if r.Method == "GET" {

renderHtml(w, "upload.html", nil)

}

if r.Method == "POST" {

f, h, err := r.FormFile("image")

//log.Println(f, h, err)

check(err)

filename := h.Filename

defer f.Close()

//log.Println(UPLOAD_DIR, filename)

t, err := ioutil.TempFile(UPLOAD_DIR, filename)

//log.Println(t)

check(err)

defer t.Close()

_, err = io.Copy(t, f) //dst src

check(err)

http.Redirect(w, r, "/view?

id="+filename, http.StatusFound)





}

}

func viewHandler(w http.ResponseWriter, r *http.Request) {

imageId := r.FormValue("id")

imagePath := UPLOAD_DIR + "/" + imageId

log.Println(imageId, imagePath)

if exists := isExists(imagePath); !exists {

http.NotFound(w, r)

return

}

//log.Println("222222222222")

w.Header().Set("Content-Type", "image")

http.ServeFile(w, r, imagePath)

}

func listHandler(w http.ResponseWriter, r *http.Request) {

fileInfoArr, err := ioutil.ReadDir("./uploads")

//log.Println(fileInfoArr)

check(err)

if err != nil {

http.Error(w, err.Error(), http.StatusInternalServerError)

return

}

locals := make(map[string]interface{})

images := []string{}

for _, fileInfo := range fileInfoArr {

//log.Println(fileInfo)

images = append(images, fileInfo.Name())

}

locals["images"] = images //这里map的key是images,相应list.html里面的$.images

renderHtml(w, "list.html", locals)





}



//巧用闭包避免执行时错误崩溃

func safeHandler(fn http.HandlerFunc) http.HandlerFunc {





return func(w http.ResponseWriter, r *http.Request) {

defer func() {

log.Println("can't past execute or finished execute")

if e, ok := recover().(error); ok {

log.Println("50x error")

//50x error

http.Error(w, e.Error(), http.StatusInternalServerError)

log.Println("Warn : panic in %v. - %v", fn, e)

log.Println(string(debug.Stack()))

}

}()





log.Println("if no panic then first execute")

fn(w, r)

}

}

func staticDirHandler(mux *http.ServeMux, prefix string, staticDir string, flags int) {

mux.HandleFunc(prefix, func(w http.ResponseWriter, r *http.Request) {

file := staticDir + r.URL.Path[len(prefix)-1:]

if (flags & ListDir) == 0 {

if exists := isExists(file); !exists {

http.NotFound(w, r)

return

}

}

http.ServeFile(w, r, file)

})





}

func main() {

mux := http.NewServeMux()

staticDirHandler(mux, "/assets/", "./public", 0)

mux.HandleFunc("/", safeHandler(listHandler))

mux.HandleFunc("/view", safeHandler(viewHandler))

mux.HandleFunc("/upload", safeHandler(uploadHandler))

err := http.ListenAndServe(":8080", mux)

if err != nil {

log.Fatal("ListenAndServe:", err.Error())

}

}

uoload.html

<html>

<head>

<meta charset="utf-8">

<title>upload</title>

</head>

<body>

<form method="POST" action="/upload" enctype="multipart/form-data">

choosing an image to upload :<br>

<input name="image" type="file">

<input type="submit" value="Upload">

</form>

</body>

</html>

list.html

<html>

<head>

<meta charset="utf-8">

<title>list</title>

</head>

<body>

<ol>

{{range $.images}}

<li><a href="/view?id={{.|urlquery}}">{{.|html}}</a></li>

{{end}}

</ol>

</body>

</html>

注意文件结构

photoweb

-photoweb.go

-public

-js

-css

-images

-uploads

-views

-upload.html

-list.html

解决调用七牛音频问题

演示样例仅仅提供代码方式调用api,不可以像图片处理直接在url中加入參数进行改动

若强用url会提示要预处理音频转码。一般转码採用异步方式

watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvbmVlZGthbmU=/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center" alt="">

OK,在go的SDK代码中进行,结果发现api给的參数仅仅是个大概,go的api里面的相应參数。不全然和官网贴出来的一样

我们搜需PutPolicy

在rs/token.go 中,看到相应deadline。grep 'xx' *  -R查看文件中内容

package main





import (

"fmt"

. "github.com/qiniu/api/conf"

"github.com/qiniu/api/fop"

"github.com/qiniu/api/io"

"github.com/qiniu/api/rs"

"log"

)





func init() {





ACCESS_KEY = "自己的ak"

SECRET_KEY = "自己sk"

}





//GET upload access token

func uptoken(bucketName string) string {

putPolicy := rs.PutPolicy{

Scope: bucketName,

//CallbackUrl: callbackUrl,

//CallbackBody:callbackBody,

//ReturnUrl: returnUrl,

//ReturnBody: returnBody,

//AsyncOps: asyncOps,

//EndUser: endUser,

//Expires: expires,

Expires:             1406555272, //截止时间戳

PersistentOps:       "avthumb/mp3",

PersistentNotifyUrl: "http://fake.com/qiniu/notify",

}

return putPolicy.Token(nil)

}





func main() {

//上传本地文件

upload("a")





//5.1 获取文件信息

//getFileInfo()





//6.1.1 查看图像属性

//imageAttr()





//5.2 删除文件

//delFile()





}





//6.1.1 查看图像属性

func imageAttr() {

var imageUrl = "http://needkane.qiniudn.com/kane2.jpg"

ii := fop.ImageInfo{}

infoRet, err := ii.Call(nil, imageUrl)

if err != nil {

// 产生错误

log.Println("fop getImageInfo failed:", err)

return

}





log.Println(infoRet.Height, infoRet.Width, infoRet.ColorModel, infoRet.Format)

}





func makeImageInfoUrl(imageUrl string) string {

ii := fop.ImageInfo{}

return ii.MakeRequest(imageUrl)

}





//5.2 删除文件

func delFile() {

bucket := "needkane"

key := "goupload.jpg"

var rsCli = rs.New(nil)





err := rsCli.Delete(nil, bucket, key)

if err != nil {

// 产生错误

log.Println("rs.Copy failed:", err)

return

}

}





//5.1 获取文件信息

func getFileInfo() {

var ret rs.Entry

bucket := "needkane"

key := "kane3.jpg"

var rsCli = rs.New(nil)

var err error

ret, err = rsCli.Stat(nil, bucket, key)





if err != nil {

// 产生错误

log.Println("rs.Stat failed:", err)

return

}





// 处理返回值

log.Println(ret)

}





//上传本地文件

func upload(key string) {

uptoken := uptoken("needkane")

fmt.Printf("uptoken:%s\n", uptoken)





var err error

var ret io.PutRet

var extra = &io.PutExtra{

//Params: params,

//MimeType: mieType,

//Crc32: crc32,

//CheckCrc: CheckCrc,

}





var localFile = "/home/qboxtest/Downloads/a.wav"





// ret 变量用于存取返回的信息,详情见 io.PutRet

// uptoken 为业务server生成的上传口令

// key 为文件存储的标识

// localFile 为本地文件名称

// extra 为上传文件的额外信息。详情见 io.PutExtra,可选

err = io.PutFile(nil, &ret, uptoken, key, localFile, extra)





if err != nil {

//上传产生错误

log.Print("io.PutFile failed:", err)

return

}





//上传成功,处理返回值

log.Print(ret.Hash, ret.Key)





}

用 file 命令查看文件

第一个go的web程序;调用七牛云存储的音频api问题解决;条件搜寻文件中的内容,字符串拼接+在上一行的更多相关文章

  1. ueditor上传图片到七牛云存储(form api,java)

    转:http://my.oschina.net/duoduo3369/blog/174655 ueditor上传图片到七牛云存储 ueditor结合七牛传图片 七牛的试炼 开发前的准备与注意事项说明 ...

  2. 七牛云存储的 Javascript Web 前端文件上传

    因为我的个人网站 restran.net 已经启用,博客园的内容已经不再更新.请访问我的个人网站获取这篇文章的最新内容,七牛云存储的 Web 前端文件上传 七牛是不错的云存储产品,特别是有免费的配额可 ...

  3. 代码实现:定义一个文件输入流,调用read(byte[] b)方法,将a.txt文件中的内容打印出来(byte数组大小限制为5)

    package com.loaderman.test; import java.io.ByteArrayOutputStream; import java.io.FileInputStream; im ...

  4. CodeIgniter - 集成七牛云存储

    最近有一个项目需要集成七牛云存储的图片存储和调用功能,程序是基于CodeIgniter2.1.3的PHP框架.刚拿到手完全无从下手的感觉,因为像框架这种东西,想从官方的PHPSDK集成进去,需要改动很 ...

  5. Go语言实战 - 网站性能优化第一弹“七牛云存储”

    由于用户纷纷反应山坡网的打开速度比较慢,所以两天前我们决定把服务器从linode迁移到阿里云. 整个迁移过程非常平滑,基本上一个小时就完成了.而且阿里云的配套设施提供的也很不错,运行状态监控什么的都有 ...

  6. JavaWeb结合七牛云存储搭建个人相册服务

    JavaWeb结合七牛云存储搭建个人相册服务 一.引言1. 课程概述 相信很多人都知道网站一般会有很多图片,对于小型网站来说,图片放在网站服务器上不算什么,但当图片数量很大时,会造成服务器很臃肿,相应 ...

  7. 《介绍一款开源的类Excel电子表格软件》续:七牛云存储实战(C#)

    两个月前的发布的博客<介绍一款开源的类Excel电子表格软件>引起了热议:在博客园有近2000个View.超过20个评论. 同时有热心读者电话咨询如何能够在SpreadDesing中实现存 ...

  8. PHP使用七牛云存储之图片的上传、下载、303重定向教程,CI框架实例

    网上关于七牛云存储的教程除了官网上的API文档,其他的资料太少了.研究了下API之后,现在已经能实现图片的上传和下载及上传之后的重定向. http://blog.csdn.net/cqcre/arti ...

  9. wordpress使用video.js与七牛云存储实现无广告视频分享应用

    video.js是一款极受欢迎的基于HTML5的开源WEB视频播放器,其充分利用了HTML5的视频支持特性,可以实现全平台的无视频插件播放功能,对于现在流行的手机.PAD等移动智能终端有极佳的应用体验 ...

随机推荐

  1. SQL Server:使用 PIVOT 行转列和 UNPIVOT 列转行

    ylbtech-SQL Server:使用 PIVOT 行转列和 UNPIVOT 列转行 可以使用 PIVOT 和 UNPIVOT 关系运算符将表值表达式更改为另一个表.PIVOT 通过将表达式某一列 ...

  2. Ubuntu sh命令无法正确执行 (修改默认sh为bash)

    新安装的ubuntu,执行shell命令,都怀疑自己的shell水平了. 原来, ubuntu默认的是dash, 需要手动执行命令将dash改为bash. 命令: sudo dpkg-reconfig ...

  3. iOS开发-UIButton浅谈

    UIButton算是最基本的一个控件了,不过有的时候用法挺多关于UIButton文字的位置,字体大小,字体的颜色 1.设置UIButton字体大小,尤其注意不要使用直接调用setFont: [self ...

  4. 大数据开发实战:Stream SQL实时开发一

    1.流计算SQL原理和架构 流计算SQL通常是一个类SQL的声明式语言,主要用于对流式数据(Streams)的持续性查询,目的是在常见流计算平台和框架(如Storm.Spark Streaming.F ...

  5. 【LeetCode】Number of Islands

    Number of Islands 问题描写叙述 Given a 2d grid map of '1's (land) and '0's (water), count the number of is ...

  6. Android Studio中实现AIDL

    AIDL 先来两个传送门: http://www.cnblogs.com/yydcdut/p/3961545.html Android面试,与Service交互方式 http://www.cnblog ...

  7. 什么是JSP (转)

    http://blog.csdn.net/javaloveiphone/article/details/7937170 一.什么是JSP(JavaServer Pages),原来是没有jsp的,只有s ...

  8. Linq-进行Json序列化的过程中出现错误解决办法

    错误截图如下: 这是因为表t_sysuser与表t_sysrole之间存在外键联系导致的 解决办法: 进入到创建的linq to sql类中,右键[属性]-将序列化模式修改为[单向]保存即可

  9. .net平台性能很不错的轻型ORM类Dapper

    dapper只有一个代码文件,完全开源,你可以放在项目里的任何位置,来实现数据到对象的ORM操作,体积小速度快. 使用ORM的好处是增.删.改很快,不用自己写sql,因为这都是重复技术含量低的工作,还 ...

  10. 刷完OpenWrt在浏览器无法访问的解决办法

    其实问题很明显. 是因为刷了trunk版固件. 并没有集成luci. 那接下来就是装luci.但是装luci需要联网(不过其实不联网其实也是可以安装的.) 我说的联网是让路有联网.而不是网线接路由,路 ...