被老板怼后,我为uni-app项目引入环境标志

前情

最近公司在规划一个全新项目,但是又对项目前景有些怀疑,于是想做一个项目获客验证的运营活动,就是为了决定后续项目可行性和投入规模。

注:时间都宝贵,如果不想浪费时间看一些无聊的事情原委的,只想了解环境标志是怎么回事的,可直接跳到实现环境标志段即可

测试正式难区分

小程序的发包流程的,在HBuilder X上点击发行-小程序就会自动打生产包,同时唤起小程序IDE,再通过IDE点点测测后没啥问题就再通过小程序IDE的上传版本上传代码包到小程序后台,此时可以查看体验版,体验版码给到测试验收没啥问题后,再走提审流程,审核通过后就可以发到线上

我负责的前一个小程序因为因为测试和生产的配置数据相差比较大,一眼就能区分当前是在测试还是在生产,这样在上传包的时候就能轻松区分当前接口是否是在生产环境,我现在负责的小程序首页只是一个入口页,和生产是一模一样的,我平时上传包的时候是看一下目前运行的代码目录来区分是在开发还是在生产的

一坑又一坑

项目其实在上线前2天就已经测试差不太多,但卡在一个前置条件,我们活动需要依赖支付宝的芝麻免押,又因为我们的项目流程因各种原因被支付宝打回,所以项目流程一直在调整,后面为了申请免押,直到芝麻免押过审一直反复提了4版,最后一版是产品完全推倒重新做的需求稿,我在开发这一版的时候我重新牵了一个新分支,因为过审后这一版是直接废掉的,因为老板一直在催,我于加急调整需求,在开发到80%的时候,产品突然找到我说芝麻免押过审了,我现在手上的需求可以搁置,让我重新回滚到正常的那一版(掉坑1)。

支付宝小程序和别的小程序不一样,它是按迭代来做小程序版本管理的,在反复的提版中我因为有一版刚刚走完了提审迭代到了待发布状态,此时产品又提了修改点,我于是就废掉了这一迭代,调整需求后重新新提了一个迭代,走了提审,提审后,我看着迭代列表中刚刚那个没走完的迭代一直那么显眼,我看着它总感觉不爽,于是我就删除了它(掉坑2)

此次活动主入口是在抖音小程序,在抖音上下单后,通过短信携带链接发送给用户,用户点击短信唤起支付宝小程序完成余下芝麻免押流程,在走通测试后,服务端做了下正式服商品配置,因为此次活动较紧急没有做配置后台,于是服务端把测试服商品调整为了和正式库一样的配置,方便到时上线直接更新(掉坑3)

被怼

因抖音小程序活动的测试数据被服务端同步和正式库数据成了一样,当我在做最后发版验收的时候,怎么都支付不了,我和服务端二个都自查了代码,也连调了好一会也没有找到问题原因,最后还是找的抖音客服才知道是因为测试库对支付是有金额限制的,不能大太,只能是小金额,具体经过详见我的博文:抖音小程序支付错误码141211

祸不单行,在做支付宝发包的时候,不知道因为什么原因,导致支付宝小程序IDE没法自动迭代,具体经过可以查看我的博文:支付宝小程序IDE版本迭代异常

直到晚上22点多二个问题都还没有解决,而此时老板一直在问进度,说今晚一定要上线......

抖音小程序因为线上是可以支付,我们于是说先不管测试支付不了的问题,先提生产版,等审核过了直接去线上验证,有问题直接回滚

支付宝因为前面提了几版用于申请芝麻免押能力,而正式上线是要去掉这些,产品让我回滚代码到正常版,而此时我理解的是上面新拉的调整分支不用,直接回到主分支再提一版即可,其实这里是错的,因为主分支也加了一些小修改用于申请芝麻免押能力。

还有个致命问题就是支付宝小程序IDE无法上传版本,找了支付宝技术人员,他们也没有找到问题,在我多次要求能不能先不管问题,先通过别的方式先绕过这个问题先提版本再说,支付宝技术人员教我手动建迭代试试,此时已经晚上11点了,我急忙忙的去手动新建迭代,此时IDE可以下拉选择到新建迭代了,也就临时绕过问题可以发包了,同时向支付宝技术人员要求能不能走个加急审核,支付宝说可以,此时我一心只想尽快提审版本,于是乎就提了审,几分钟内就过了审,我也点了发包,终于松了一口气

但是没过几分钟产品跑过来说回退的版本是错的,带了芝麻免押审核的一些调整,又过了一会,测试又跑过来说发的版本有问题一直报订单不存在,我当时就蒙了,因为我只是切了下分支,主分支原先就是验证通过的,直到BOSS跑过来说,线上的商品价格怎么有1块的,是不是发错环境了,我此时意识到问题了,我在上传版本的时候因为紧急忘了确认当前是生产还是测试了,因当时电脑上测试和生产同时跑着,我一急就点错了,BOSS此时很不耐烦的说,你写小程序也有这么久了,怎么还会出现这种问题,一会抖音支付不了,一会又包错包,我此时尴尬到了极点,我于是加紧了提了一版,幸运提此次提版审核也十来分钟就过了,顺利的发了包

此时已经晚上11.30分,过了晚上11点就没有地铁了,我打的的士,在的士上我反复在想为什么会犯这种低级错,生产和测试包都能提错,我想到平时自己发包都是通过看代码路径来判断是否是生产和测试的,如果不急的情况下,是不会有问题的,但是事情一紧急就会忘记去确认代码路径,我此时想到应该给项目引入环境标志,一眼就能看出当前是什么环境,避免因手忙脚乱导致发错环境。

引入环境标志

环境标示-日志

想到的第一种方式是通过打印日志,在日志中标志出当前服务端地址,打印普通的日志不是十分显眼的,很容易被淹没,我于是选择打印一个定制化的彩色日志,效果如下:

关键代码如下:

jsx 复制代码
import { BASE_URL } from '@/config/http';

/**
 * 环境日志打印
 */
export const initEnvLog = () => {
  uni.$off('envlog');
  uni.$on('envlog', () => {
    console.log(`%c--- 项目环境 ----:%c${process.env.NODE_ENV} ${BASE_URL.split('//')[1]}`, 'color:white;background:blue;padding:6px;', 'color:red;background:white;padding:6px;');
  })
}

任何地方只要调用uni.$emit('envlog')即可在控制台打印环境日志

环境标志-页面标示

为什么做了日志环境标示还要做页面标示,为了开发调试需求一般都会在控制台打印非常多的日志,像我现在的项目,为了能快速发现服务端接口问题,我在封装通用接口请求的时候,默认会把请求信息和接口响应信息全打印

还有一些别的开发日志,控制台是很丰富,虽然日志标志已经做成鹤立鸡群,一眼可以看出,但还是有很大可能日志会被淹没,于是想到做一个页面标志

我把它封装成组件并全局注册,在所有页面都使用此组件即可,为了不影响测试测试,同时为该标志增加了拖动功能,可以手动依靠在页面任一位置,同时还把项目彩蛋页入口也加到上面,如果你快速点击按钮6次+,就会让你选择是隐藏标志,还是跳彩蛋页,有了页面标志,就算你一时急,没有查看日志。页面上一个大大的测试服是逃不掉你我们眼镜的,组件代码如下:

jsx 复制代码
<template>
    <image 
        class="env-log" 
        v-if="isShow"
        @click="handleClick" 
        @touchstart="handleTouchStart"
        @touchmove="handleTouchMove"
        @touchend="handleTouchEnd"
        :style="{ transform: `translate(${position.x}px, ${position.y}px)` }"
        src="" 
        mode="aspectFit" 
    />
</template>

<script setup>
    import { ref, onMounted, nextTick } from 'vue';

  const clickCount = ref(0);
  const isShow = ref(false);

    // 拖拽相关状态
    const isDragging = ref(false);
    const startTouch = ref({ x: 0, y: 0 });
    const position = ref({ x: 0, y: 0 });
    const startPosition = ref({ x: 0, y: 0 });

  // 连续点击6次隐藏环境标志
  let timer = null;
    const handleClick = () => {
        // 如果是拖拽状态,不处理点击事件
        if (isDragging.value) return;

        clickCount.value += 1;
    clearTimeout(timer);
    timer = setTimeout(() => {
      clickCount.value = 0;
    }, 1000);
        if (clickCount.value > 6) {
      uni.showModal({
        title: '',
        content: '请选择操作',
        cancelText: '隐藏标志',
        confirmText: '彩蛋',
        success: (res) => {
          if (res.confirm) {
            uni.navigateTo({
              url: '/other/egg/egg'
            })
          } else {
            isShow.value = false;
          }
        }
      })
        }
    }

    // 触摸开始
    const handleTouchStart = (e) => {
        e.preventDefault();
        isDragging.value = false;
        const touch = e.touches[0];
        startTouch.value = {
            x: touch.clientX,
            y: touch.clientY
        };
        startPosition.value = { ...position.value };
    }

            // 触摸移动
    const handleTouchMove = (e) => {
        e.preventDefault();
        if (!startTouch.value) return;

        const touch = e.touches[0];
        const deltaX = touch.clientX - startTouch.value.x;
        const deltaY = touch.clientY - startTouch.value.y;

        // 如果移动距离超过5px,认为是拖拽
        if (Math.abs(deltaX) > 5 || Math.abs(deltaY) > 5) {
            isDragging.value = true;
        }

        if (isDragging.value) {
            const newX = startPosition.value.x + deltaX;
            const newY = startPosition.value.y + deltaY;

            // 获取窗口尺寸进行边界限制
            const systemInfo = uni.getSystemInfoSync();
            const screenWidth = systemInfo.screenWidth;
            const windowHeight = systemInfo.windowHeight; // 使用窗口高度而不是屏幕高度
            const imageWidth = 121; // rpx转px大约是屏幕宽度的1/750 * 121
            const imageHeight = 96;
            const realImageWidth = (screenWidth / 750) * imageWidth;
            const realImageHeight = (screenWidth / 750) * imageHeight;

            // 边界限制
            const maxX = screenWidth - realImageWidth;
            const maxY = windowHeight - realImageHeight;

            position.value = {
                x: Math.max(0, Math.min(newX, maxX)),
                y: Math.max(0, Math.min(newY, maxY))
            };
        }
    }

    // 触摸结束
    const handleTouchEnd = (e) => {
        e.preventDefault();
        startTouch.value = { x: 0, y: 0 };

        // 延迟重置拖拽状态,避免触发点击事件
        setTimeout(() => {
            isDragging.value = false;
        }, 100);
    }

  onMounted(() => {
    if ((process && process.env && process.env.NODE_ENV === 'development') || import.meta.env.MODE === 'development') {
      isShow.value = true;

            // 初始化位置到右下角
            nextTick(() => {
                const systemInfo = uni.getSystemInfoSync();
                const screenWidth = systemInfo.screenWidth;
                const windowHeight = systemInfo.windowHeight; // 使用窗口高度而不是屏幕高度
                const imageWidth = 121;
                const imageHeight = 96;
                const realImageWidth = (screenWidth / 750) * imageWidth;
                const realImageHeight = (screenWidth / 750) * imageHeight;

                position.value = {
                    x: screenWidth - realImageWidth,
                    y: windowHeight - realImageHeight - 100 // 距离底部100px,基于窗口高度
                };
            });
    } else {
      isShow.value = false;
    }
  })

</script>

<style lang="scss">
.env-log {
    width: 121rpx;
    height: 96rpx;
  position: fixed;
  top: 0;
  left: 0;
  opacity: 0.65;
  z-index: 9999;
    transition: none; // 拖拽时不要过渡动画
    user-select: none; // 防止选中
}
</style>

引入的环境标志效果如下:

期望

有的人一定会说为了一个判断环境特意封一个组件,有点杀鸡用牛刀,解决问题的方法有千千万,能解决你当下问题的方法,就是好的方法

我期望的做法的,我想后续封装一个vite插件,通过插件注入日志组件,生产环境直接丢掉,这样不会为生产带去多余的代码同时也是一劳永逸的事,不像现在还要手动插入

如果你有更棒的解决方案,欢迎你留言分享,一起学习进步

相关推荐
sasaraku.30 分钟前
serviceWorker缓存资源
前端
RadiumAg2 小时前
记一道有趣的面试题
前端·javascript
yangzhi_emo2 小时前
ES6笔记2
开发语言·前端·javascript
yanlele2 小时前
我用爬虫抓取了 25 年 5 月掘金热门面试文章
前端·javascript·面试
The_era_achievs_hero2 小时前
微信小程序71~80
微信小程序·小程序
中微子3 小时前
React状态管理最佳实践
前端
米粒宝的爸爸3 小时前
uniapp在app端,在导航栏设置自定义按钮
uni-app
烛阴3 小时前
void 0 的奥秘:解锁 JavaScript 中 undefined 的正确打开方式
前端·javascript
中微子3 小时前
JavaScript 事件与 React 合成事件完全指南:从入门到精通
前端
Hexene...3 小时前
【前端Vue】如何实现echarts图表根据父元素宽度自适应大小
前端·vue.js·echarts