npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2025 – Pkg Stats / Ryan Hefner

@theracerx/custom-google-crud

v0.0.0

Published

<h1>Custom Google CRUD</h1> <p> This is a simplified CRUD program that mimics backend development. It uses spreadsheet instead of databases.The functions were categorized into 4 parts: ACTIONS, GET, POST, DELETE. </p>

Downloads

8

Readme

//for GET HTTP method // can be GET only // for customer viewing //return ContentService.createTextOutput is only sent back to requestor in this function!!! //never use ASYNC function doGet(e) { Logger.log("doGet started");

let res; //TEST GET sheet //res = GET.query({sheet:ss.getSheetByName("Sheet1")}) //return Sheet

//TEST GET row //res = GET.query({sheet:ss.getSheetByName("Sheet1"),rowPos:1}) //return Row

//TEST GET column //res = GET.query({sheet:ss.getSheetByName("Sheet1"),colPos:1}) //return Column

//TEST GET cell //res = GET.query({sheet:ss.getSheetByName("Sheet1"),colPos:1,rowPos:1}) //return Cell

//TEST GET a1notation //res = GET.query({sheet:ss.getSheetByName("Sheet1"),a1notation:"A1"}) //return Cell

//TEST GET r1c1notation //res = GET.query({sheet:ss.getSheetByName("Sheet1"),r1c1notation:"A1:B4"}) //return Range

//TEST GET find matching columnVal given rowVal //res = GET.query({sheet:ss.getSheetByName("Sheet1"),field1:"Link",field2:"Year",rowVal:"asdf"}) //missing fiedrowpos //res = GET.query({sheet:ss.getSheetByName("Sheet1"),field1:"Year",field2:"Link",fieldRowPos:1,rowVal:"2024"}) //searches to row1 //res = GET.query({sheet:ss.getSheetByName("Sheet1"),field1:"Link",field2:"Year",fieldRowPos:1,rowVal:"asdf"}) //searches to row1

Logger.log(res) if(res == "TextOutput"){ return res } else return MISC.sendStatus(true,res) }

//for POST HTTP method // can be PATCH, POST, or DELETE // for editors only function doPost(e) { Logger.log("doPost started");

//TEST POST sheet
//res = POST.setValue({sheet:ss.getSheetByName("Sheet2"),values:[["A","B","C"]]}) //will Error

//TEST POST row
//res = POST.setValue({sheet:ss.getSheetByName("Sheet2"),rowPos:2,values:[["A","B","C"]],length:3}) //return Row

//TEST POST column
//res = POST.setValue({sheet:ss.getSheetByName("Sheet2"),colPos:2,values:[["A"],["B"],["C"]],length:3}) //return Column

//TEST POST cell
//res = POST.setValue({sheet:ss.getSheetByName("Sheet2"),colPos:3,rowPos:3,value:"AAA"})  //return Cell

//TEST POST a1notation
//res = POST.setValue({sheet:ss.getSheetByName("Sheet2"),a1notation:"A1",value:"TEST"})  //return Cell

//TEST POST r1c1notation
//res = POST.setValue({sheet:ss.getSheetByName("Sheet2"),r1c1notation:"A1:B2",values:[[1,2],[1,2]]})  //return Range

//TEST POST find matching columnVal given rowVal
//res = POST.setValue({sheet:ss.getSheetByName("Sheet2"),field1:"Link",field2:"Year",rowVal:"asdf"}) //missing fiedrowpos
//res = POST.setValue({sheet:ss.getSheetByName("Sheet2"),field1:"a",field2:"b",fieldRowPos:1,rowVal:"a1",value:"GGG"}) //searches to row1
//res = POST.setValue({sheet:ss.getSheetByName("Sheet2"),field1:"Link",field2:"Year",fieldRowPos:1,rowVal:"asdf"},true) //searches to row1

} var ACTION = {} //add your custom actions here var GET = {//default GET //QUERY commands-------------------------------------------- //length is only used by POST & PATCH for querying columns and rows query:({sheet,rowPos,colPos,rowVal,colVal,field1,field2,fieldRowPos,a1notation,r1c1notation,length},returnSrc)=>{ //shorthand command
Logger.log("query started")

  //min req a1notation
  if(a1notation){
    return GET.queryA1Notation({sheet:sheet,a1notation:a1notation},returnSrc)
  }

  //min req r1c1notation
  if(r1c1notation){
    return GET.queryR1C1Notation({sheet:sheet,r1c1notation:r1c1notation},returnSrc)
  }

  // min req rowPos, colPos
  if(!rowPos && colPos){ //returns column
    return GET.queryColumn({sheet:sheet,colPos:colPos,length:length},returnSrc)
  } else if(!colPos && rowPos){ //returns row
    return GET.queryRow({sheet:sheet,rowPos:rowPos,length:length},returnSrc)
  } else if(rowPos && colPos){//returns value at x,y
    return GET.queryCell({sheet:sheet,rowPos:rowPos,colPos},returnSrc)
  }

  // min req field1, field2, fieldRowPos, colVal | rowVal
  //give X1,Y1,Y2 find X2 (i.e. X1 = 3; Y1 = ID; Y2 = NAME; then X2 must be Pig)
  if(field1 && field2){ //return matching value given colVal1
    //colPos, colVal,
    return GET.queryMatchCell({sheet:sheet,rowVal:rowVal,colVal:colVal,field1:field1,field2:field2,fieldRowPos:fieldRowPos},returnSrc)
  } else if (!field1 && field2){ //error
    return MISC.sendStatus(false,"cannot query with unknown field2")
  } else if (field1 && !field2){ //error
    return MISC.sendStatus(false,"cannot query with unknown field2")
  }

  return GET.querySheet({sheet:sheet},returnSrc)
},


//QUERY SUPPORT COMMAND
queryCell:({sheet,rowPos,colPos},returnSrc)=>{
  if(sheet != "Sheet"){
    return MISC.sendStatus("false","cannot query sheet not instance of Sheet")
  }
  if (!rowPos) {
    return MISC.sendStatus(false, "cannot queryCell unknown `rowPos`");
  }
  if (!colPos) {
    return MISC.sendStatus(false, "cannot queryCell unknown `colPos`");
  }

  if(!returnSrc){return sheet.getRange(rowPos,colPos).getValue() 
  } else return sheet.getRange(rowPos,colPos)
},
queryMatchCell:({sheet,rowVal,colVal,field1,field2,fieldRowPos},returnSrc)=>{
  let findMatch = ()=>{ //returns query with rowPos colPos
      if(!fieldRowPos)return MISC.sendStatus(false,"cannot query with fieldRowPos = undefined")

      //Logger.log("fieldRowPos:" + fieldRowPos)
      //Logger.log("rowVal:" + rowVal)
      //Logger.log("colVal:" + colVal)
      //Logger.log(rowVal || colVal)


      let trgtColPos1 = GET.getColPosInRow({sheet:sheet,rowPos:fieldRowPos,rowVal:field1})
      Logger.log("trgtColPos1")
      Logger.log(trgtColPos1 == "TextOutput")
      if(trgtColPos1 == "TextOutput"){

      }
      let trgtRowPos = GET.getRowPosInCol({sheet:sheet,colPos:trgtColPos1,colVal:rowVal || colVal})
      let trgtColPos2 = GET.getColPosInRow({sheet:sheet,rowPos:fieldRowPos,rowVal:field2})

      return GET.query({sheet:sheet,rowPos:trgtRowPos,colPos:trgtColPos2},returnSrc)
    }

    if(rowVal && !colVal){ return findMatch()
    } else if(!rowVal && colVal){ return findMatch()
    } else if(rowVal && colVal && rowVal == colVal){ return findMatch()
    } else if(rowVal && colVal && rowVal != colVal){
      return MISC.sendStatus(false,"cannot query with both rowVal & colVal present")
    } else if (!rowVal && !colVal){
      return MISC.sendStatus(false,"cannot query with neither rowVal & colVal present")
    }
},
getColPosInRow:({sheet,rowPos,rowVal})=>{
  Logger.log("getColPosInRow started")
  //verify args if present

  if(sheet != "Sheet"){
    return MISC.sendStatus("false","cannot query sheet not instance of Sheet")
  }

  if (!rowPos) {
    return MISC.sendStatus(false, "cannot getColPosInRow unknown `rowPos`");
  }
  if (!rowVal) {
    return MISC.sendStatus(false, "cannot getColPosInRow unknown `rowVal`");
  }

  var pos = null;
  var lstCol = sheet.getLastColumn();
  var collection = []
  for (var i = 1; i <= lstCol; i++) {
      let val = sheet.getRange(rowPos, i).getValue()
      if (val == rowVal) {
          pos = i;
          break;
      } else collection.push(val)
  }
  if (pos == null) {
    Logger.log("getColPosInRow failed: " + rowVal + " is missing from sheet:" + sheet.getName() + " ,row:" + collection)
    return MISC.sendStatus(false, "cannot query rowVal: " + rowVal + " doesn't exist in sheet[" + sheet.getName() + "]");
  } else {
    Logger.log("getColPosInRow ended: " + pos)
    return pos
  }
},
getRowPosInCol:({sheet,colPos,colVal})=>{
  Logger.log("getRowPosInCol started")
  //verify args if present

  if(sheet != "Sheet"){
    return MISC.sendStatus("false","cannot query sheet not instance of Sheet")
  }
  if (!colPos) {
    return MISC.sendStatus(false, "cannot getRowPosInCol unknown `rowPos`");
  }
  if (!colVal) {
    return MISC.sendStatus(false, "cannot getRowPosInCol unknown `rowVal`");
  }

  var pos = null;
  var lstRow = sheet.getLastRow();
  var collection = []
  for (var i = 1; i <= lstRow; i++) {
    let val = sheet.getRange(i, colPos).getValue()
      if (val == colVal) {
          pos = i;
          break;
      } else collection.push(val)
  }
  if (pos == null) {
    Logger.log("getRowPosInCol failed: " + colVal + " is missing from sheet:" + sheet.getName() + " ,column:" + collection)
    return MISC.sendStatus(false, "cannot query colVal: " + colVal + " doesn't exist in sheet[" + sheet.getName() + "]");
  } else {
    Logger.log("getRowPosInCol ended: " + pos )
    return pos
  }
},
queryRow:({sheet, rowPos, length},returnSrc)=>{
  Logger.log("queryRow started")

  if(sheet != "Sheet"){
    return MISC.sendStatus("false","cannot query sheet not instance of Sheet")
  }
  if(!rowPos){
      return MISC.sendStatus(false, "cannot queryRow unknown `rowPos`");
  }

  if(!returnSrc){return sheet.getRange(rowPos, 1, 1, length || sheet.getLastColumn()).getValues();
  } else return sheet.getRange(rowPos, 1, 1, length || sheet.getLastColumn())
},
queryColumn:({sheet, colPos, length},returnSrc)=>{
  Logger.log("queryColumn started")

  if(sheet != "Sheet"){
    return MISC.sendStatus("false","cannot query sheet not instance of Sheet")
  }
  if(!colPos){
      return MISC.sendStatus(false, "cannot queryColumn unknown `colPos`");
  }

  if(!returnSrc){return sheet.getRange(1,colPos,length ||sheet.getLastRow(),1).getValues();
  } else return sheet.getRange(1,colPos,length ||sheet.getLastRow(),1);
},
querySheet:({sheet},returnSrc)=>{
  Logger.log("querySheet started")

  if(sheet != "Sheet"){
    return MISC.sendStatus("false","cannot query sheet not instance of Sheet")
  }

  if(!returnSrc){return sheet.getRange(1,1,sheet.getLastRow()|| 1,sheet.getLastColumn()|| 1).getValues();
  } else return sheet.getRange(1,1,sheet.getLastRow() || 1,sheet.getLastColumn() || 1)
},
queryA1Notation:({sheet,a1notation},returnSrc)=>{
  Logger.log("queryA1Notation started")

  if(sheet != "Sheet"){
    return MISC.sendStatus("false","cannot query sheet not instance of Sheet")
  }

  if(!a1notation){
    return MISC.sendStatus(false, "cannot queryA1Notation unknown a1notation");
  }

  if(!returnSrc){return sheet.getRange(a1notation).getValue() 
  } else return sheet.getRange(a1notation)
},
queryR1C1Notation:({sheet,r1c1notation},returnSrc)=>{
  Logger.log("queryR1C1Notation started")

  if(sheet != "Sheet"){
    return MISC.sendStatus("false","cannot query sheet not instance of Sheet")
  }

  if(!r1c1notation){
    return MISC.sendStatus(false, "cannot queryR1C1Notation unknown r1c1notation");
  }

  if(!returnSrc){return sheet.getRange(r1c1notation).getValues() 
  } else return sheet.getRange(r1c1notation)
},

}; var POST = {//default POST // SET VALUE Commands ---------------------------------------

  //A1:C1
  //[[A1,B1,C1]]
  //A1:A5 like this
  //[[A1],[A2],[A3],[A4],[A5]]
  //And A1:C2 like this
  //[[A1,B1,C1],[A2,B2,C2]]
 
setValue:({sheet,rowPos,colPos,rowVal,colVal,field1,field2,fieldRowPos,a1notation,r1c1notation,value,values,length})=>{
  Logger.log("setValue started")

  if(sheet != "Sheet"){
    return MISC.sendStatus("false","cannot query sheet not instance of Sheet")
  }

  if(value){
    if(Array.isArray(value)){
      return MISC.sendStatus(false,"cannot setValue with single value instance of Array")
    }

    //min req a1notation
    if(a1notation){
      return POST.setA1notationVal({sheet:sheet,a1notation:a1notation,value:value})
    }
    
    if(rowPos && colPos){//returns value at x,y
      return POST.setCellVal({sheet:sheet,rowPos:rowPos,colPos,value:value})
    }
    
    // min req field1, field2, fieldRowPos, colVal | rowVal
    //give X1,Y1,Y2 find X2 (i.e. X1 = 3; Y1 = ID; Y2 = NAME; then X2 must be Pig)
    if(field1 && field2){ //return matching value given colVal1
      //colPos, colVal,
      return POST.setMatchCellVal({sheet:sheet,rowVal:rowVal,colVal:colVal,field1:field1,field2:field2,fieldRowPos:fieldRowPos,value:value})
    } else if (!field1 && field2){ //error
      return MISC.sendStatus(false,"cannot setValue with unknown field2")
    } else if (field1 && !field2){ //error
      return MISC.sendStatus(false,"cannot setValue with unknown field2")
    }

  } else if (values){

    if(!Array.isArray(values)){
      return MISC.sendStatus(false,"cannot setValue with values not an instance of Array")
    }

    //min req r1c1notation
    if(r1c1notation){
      return POST.setR1C1NotationVal({sheet:sheet,r1c1notation:r1c1notation,values:values})
    }

    // min req rowPos, colPos
    if(!rowPos && colPos){ //returns column
      return POST.setColumnVal({sheet:sheet,colPos:colPos,values:values,length})
    } else if(!colPos && rowPos){ //returns row
      return POST.setRowVal({sheet:sheet,rowPos:rowPos,values:values,length})
    }

    return POST.setSheetVal({sheet:sheet,values:values})

  } else  return MISC.sendStatus(false,"cannot setValue with unknown value")

},

//setValues
setR1C1NotationVal:({sheet:sheet,r1c1notation:r1c1notation,values:values})=>{
  Logger.log("setR1C1NotationVal started")
  return GET.queryR1C1Notation({sheet,r1c1notation},true).setValues(values)
},
setColumnVal:({sheet,colPos,values,length})=>{//values: [[1],[2],[3]]
  Logger.log("setColumnVal started")
  return GET.queryColumn({sheet:sheet,colPos:colPos,length},true).setValues(values)
},
setRowVal:({sheet,rowPos,values,length})=>{ //values: [[1,2,3]]
  Logger.log("setRowVal started")
  return GET.queryRow({sheet:sheet,rowPos:rowPos,length:length},true).setValues(values)
},
setSheetVal:()=>{
  Logger.log("setSheetVal started")
  return MISC.sendStatus("false","setSheetVal doesn't work, pls see dev comment")
    //obselete error
    //CANNOT DO setSheetVal because empty sheet can only be [[]]
    //to setSheetVal... an existing value must be present to be overwritten
    //but since this is a POST method... no overwritting is permitted
    //just do a series of the other command to replicate setSheetVal func
  
},

//setValue
setA1notationVal:({sheet,a1notation,value})=>{
  Logger.log("setA1notationVal started")
  return GET.queryA1Notation({sheet:sheet,a1notation:a1notation},true).setValue(value)
},
setCellVal:({sheet,rowPos,colPos,value})=>{
  Logger.log("setCellVal started")
  return GET.queryCell({sheet:sheet,rowPos:rowPos,colPos:colPos},true).setValue(value)
},
setMatchCellVal:({sheet,rowVal,colVal,field1,field2,fieldRowPos,value})=>{
  Logger.log("setMatchCellVal started")
  return GET.queryMatchCell({sheet:sheet,rowVal:rowVal,colVal:colVal,field1:field1,field2:field2,fieldRowPos:fieldRowPos},true).setValue(value)
},

};